sl@0
|
1 |
// Copyright (c) 2006-2010 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
//
|
sl@0
|
13 |
// Description:
|
sl@0
|
14 |
//
|
sl@0
|
15 |
|
sl@0
|
16 |
#ifndef __SQLBUFFLAT_H__
|
sl@0
|
17 |
#define __SQLBUFFLAT_H__
|
sl@0
|
18 |
|
sl@0
|
19 |
#include <e32base.h>
|
sl@0
|
20 |
#include "SqlAssert.h"
|
sl@0
|
21 |
|
sl@0
|
22 |
//Forward declarations
|
sl@0
|
23 |
class TSqlBufRIterator;
|
sl@0
|
24 |
class CSqlSecurityPolicy;
|
sl@0
|
25 |
class TSqlSecurityPolicyIterator;
|
sl@0
|
26 |
|
sl@0
|
27 |
#ifdef _DEBUG
|
sl@0
|
28 |
const TUint64 KSqlBufFlatMagicValue = 0xAA11BB44221199EEULL;//TInt64 for 8-byte alignment
|
sl@0
|
29 |
#define SQLFLATBUF_INVARIANT() Invariant()
|
sl@0
|
30 |
#else
|
sl@0
|
31 |
#define SQLFLATBUF_INVARIANT() void(0)
|
sl@0
|
32 |
#endif
|
sl@0
|
33 |
|
sl@0
|
34 |
/**
|
sl@0
|
35 |
RSqlBufFlat class manages a flat memory buffer with fixed elements count.
|
sl@0
|
36 |
Each element (or field) of the flat buffer has a type and variable length data and is accessible using an
|
sl@0
|
37 |
index. The data may be NULL.
|
sl@0
|
38 |
The RSqlBufFlat objects are used for sending/receiving data between client dlls and servers,
|
sl@0
|
39 |
because once the buffer filled, a pointer to the flat buffer data can be used to send the whole buffer in a single
|
sl@0
|
40 |
IPC call, which is more effective than using a stream like transfer.
|
sl@0
|
41 |
|
sl@0
|
42 |
RSqlBufFlat public functions are not very convenient for storing/retrieving values based on the column type
|
sl@0
|
43 |
and there are two additional classes which may be used instead of working with the flat buffer directly:
|
sl@0
|
44 |
TSqlBufRIterator and TSqlBufWIterator.
|
sl@0
|
45 |
|
sl@0
|
46 |
The physical structure of the flat buffer is:
|
sl@0
|
47 |
---------------------------
|
sl@0
|
48 |
| SysData | Header | Data |
|
sl@0
|
49 |
---------------------------
|
sl@0
|
50 |
- "SysData" has fixed length and contains fields with about the buffer used size, elements count, header size;
|
sl@0
|
51 |
- "Header" has fixed length, which depends of the number of elements and contains cells, one per element.
|
sl@0
|
52 |
Each cell contains information about: element type, element data length, the start position of the
|
sl@0
|
53 |
element data in the flat buffer, "present"/"not present" flag. "Not present" means that no memory
|
sl@0
|
54 |
is reserved for the element data, but the buffer "knows" what is the element type and element data length;
|
sl@0
|
55 |
(In the current design there is no "Present"/"Not present flag". If the data position is 0, that indicates
|
sl@0
|
56 |
the element is "Not present"). See RSqlBufFlat::TCell comments for details.
|
sl@0
|
57 |
- "Data" Is the dynamic part of the buffer which may grow during the operations with the flat buffer (setting
|
sl@0
|
58 |
element data);
|
sl@0
|
59 |
|
sl@0
|
60 |
"Present"/"Not present" attribute has a key role in the large data transfers between the client and the server.
|
sl@0
|
61 |
In order to optimize memory usage, usually the client specifies what can be the max field size. The server
|
sl@0
|
62 |
knows that and when it fills a particular flat buffer, which has to be sent to the client, it puts in the
|
sl@0
|
63 |
flat buffer only the pieces of data whose size is less than the specified "large data" size. Any data piece whose
|
sl@0
|
64 |
size is larger will be set as "Not present" in the flat buffer. The "Not present" field will have all the
|
sl@0
|
65 |
attributes, like type and size, but won't have any data or any allocated flat buffer memory.
|
sl@0
|
66 |
So the client, when it receives the flat buffer, will know the field type and size and if it wants to get the field data,
|
sl@0
|
67 |
will have to make an additional call to the server.
|
sl@0
|
68 |
|
sl@0
|
69 |
Note that the fields in the buffer are 8-byte aligned.
|
sl@0
|
70 |
|
sl@0
|
71 |
Typical examples how the buffer can be used:
|
sl@0
|
72 |
|
sl@0
|
73 |
Case 1 - a dll client wants to receive from the server a flat buffer data.
|
sl@0
|
74 |
|
sl@0
|
75 |
@code
|
sl@0
|
76 |
RSqlBufFlat bufFlat;
|
sl@0
|
77 |
bufFlat.SetCount(N); //N is the field count
|
sl@0
|
78 |
....
|
sl@0
|
79 |
bufFlat.Reset(); //Clears the content of the buffer without shrinking it.
|
sl@0
|
80 |
ipcArgs.Set(0, bufFlat.MaxSize()); //Tell the server what is the flat buffer max size
|
sl@0
|
81 |
ipcArgs.Set(1, &bufFlat.BufPtr());
|
sl@0
|
82 |
TInt rc = session.SendReceive(funcNum, ipcArgs);
|
sl@0
|
83 |
if(rc > KSqlClientBufOverflowCode)
|
sl@0
|
84 |
{ //the client buffer is not big enough and has to be resized
|
sl@0
|
85 |
rc = bufFlat.ReAlloc(err - KSqlClientBufOverflowCode);
|
sl@0
|
86 |
if(rc != KErrNone)
|
sl@0
|
87 |
{
|
sl@0
|
88 |
return rc;
|
sl@0
|
89 |
}
|
sl@0
|
90 |
ipcArgs.Set(0, bufFlat.MaxSize());
|
sl@0
|
91 |
ipcArgs.Set(1, &bufFlat.BufPtr());
|
sl@0
|
92 |
rc = session.SendReceive(funcNum, ipcArgs);
|
sl@0
|
93 |
}
|
sl@0
|
94 |
@endcode
|
sl@0
|
95 |
|
sl@0
|
96 |
Case 2 - a dll client wants to send to the server a flat buffer data.
|
sl@0
|
97 |
|
sl@0
|
98 |
@code
|
sl@0
|
99 |
RSqlBufFlat bufFlat;
|
sl@0
|
100 |
bufFlat.SetCount(N); //N is the field count
|
sl@0
|
101 |
....
|
sl@0
|
102 |
TPtrC8 ptr(bufFlat.BufDes());
|
sl@0
|
103 |
TInt err = session.SendReceive(funcNum, TIpcArgs(ptr.Length(), &ptr));
|
sl@0
|
104 |
@endcode
|
sl@0
|
105 |
|
sl@0
|
106 |
Case 3 - the server wants to return to the client a flat buffer data.
|
sl@0
|
107 |
@code
|
sl@0
|
108 |
RSqlBufFlat bufFlat;
|
sl@0
|
109 |
bufFlat.SetCount(N); //N is the field count
|
sl@0
|
110 |
....
|
sl@0
|
111 |
TInt maxCliBufSize = msg.Int0(); //The max size of the client buffer
|
sl@0
|
112 |
if(maxCliBufSize < bufFlat.Size())
|
sl@0
|
113 |
{
|
sl@0
|
114 |
return bufFlat.Size() + KSqlClientBufOverflowCode;//Tell the client that its buffer is too small
|
sl@0
|
115 |
}
|
sl@0
|
116 |
msg.WriteL(1, bufFlat.BufDes());
|
sl@0
|
117 |
@endcode
|
sl@0
|
118 |
|
sl@0
|
119 |
Case 4 - the server wants to receive from the client a flat buffer data.
|
sl@0
|
120 |
@code
|
sl@0
|
121 |
RSqlBufFlat bufFlat;
|
sl@0
|
122 |
bufFlat.SetCount(N); //N is the field count
|
sl@0
|
123 |
....
|
sl@0
|
124 |
TInt cliBufFlatLen = aMessage.Int0();
|
sl@0
|
125 |
TInt err = bufFlat.ReAlloc(cliBufFlatLen); //Reallocate memory for the flat buffer
|
sl@0
|
126 |
if(err != KErrNone)
|
sl@0
|
127 |
{
|
sl@0
|
128 |
return err;
|
sl@0
|
129 |
}
|
sl@0
|
130 |
msg.ReadL(1, bufFlat.BufPtr());
|
sl@0
|
131 |
@endcode
|
sl@0
|
132 |
|
sl@0
|
133 |
Case 5 - the server (or the client) wants to fill the flat buffer with some data.
|
sl@0
|
134 |
@code
|
sl@0
|
135 |
RSqlBufFlat bufFlat;
|
sl@0
|
136 |
bufFlat.SetCount(N); //N is the field count
|
sl@0
|
137 |
....
|
sl@0
|
138 |
TInt err = flatBuf.SetCount(M); //If the field count has to be changed to M
|
sl@0
|
139 |
if(err != KErrNone)
|
sl@0
|
140 |
{
|
sl@0
|
141 |
return err;
|
sl@0
|
142 |
}
|
sl@0
|
143 |
//use the TSqlBufWIterator iterator to fill the buffer
|
sl@0
|
144 |
@endcode
|
sl@0
|
145 |
|
sl@0
|
146 |
@see TSqlBufRIterator
|
sl@0
|
147 |
@see TSqlBufWIterator
|
sl@0
|
148 |
@see RSqlBufFlat::TCell
|
sl@0
|
149 |
@see RSqlBufFlat::TBufFlat
|
sl@0
|
150 |
|
sl@0
|
151 |
@internalComponent
|
sl@0
|
152 |
*/
|
sl@0
|
153 |
class RSqlBufFlat
|
sl@0
|
154 |
{
|
sl@0
|
155 |
friend class TSqlBufRIterator;
|
sl@0
|
156 |
friend class CSqlSecurityPolicy;
|
sl@0
|
157 |
friend class TSqlSecurityPolicyIterator;
|
sl@0
|
158 |
|
sl@0
|
159 |
public:
|
sl@0
|
160 |
//This enum has to be in the "public" section because it is used by TCell (declared as "private" in RSqlBufFlat)
|
sl@0
|
161 |
enum
|
sl@0
|
162 |
{
|
sl@0
|
163 |
EWidthType = 3, //Bit width of the "Type" field of the header cell
|
sl@0
|
164 |
EWidthLen = 29 //Bit width of the "Length" field of the header cell
|
sl@0
|
165 |
};
|
sl@0
|
166 |
//This enum has to be in the "public" section because it is used by TCell (declared as "private" in RSqlBufFlat)
|
sl@0
|
167 |
enum
|
sl@0
|
168 |
{
|
sl@0
|
169 |
EMaxType = 1 << EWidthType, //Max allowed flat buffer field type
|
sl@0
|
170 |
EMaxLength = 1 << EWidthLen //Max allowed flat buffer field length
|
sl@0
|
171 |
};
|
sl@0
|
172 |
|
sl@0
|
173 |
RSqlBufFlat();
|
sl@0
|
174 |
TInt SetCount(TInt aCount);
|
sl@0
|
175 |
TInt ReAlloc(TInt aSize);
|
sl@0
|
176 |
void ResetAndMinimize();
|
sl@0
|
177 |
void Reset();
|
sl@0
|
178 |
void Close();
|
sl@0
|
179 |
inline TInt Count() const;
|
sl@0
|
180 |
inline TInt Size() const;
|
sl@0
|
181 |
inline TInt MaxSize() const;
|
sl@0
|
182 |
|
sl@0
|
183 |
TInt SetField(TInt aIndex, TInt aType, const void* aData, TInt aDataLength);
|
sl@0
|
184 |
|
sl@0
|
185 |
inline const TDesC8& BufDes() const;
|
sl@0
|
186 |
inline TPtr8& BufPtr();
|
sl@0
|
187 |
|
sl@0
|
188 |
private: //Data type declarations
|
sl@0
|
189 |
/**
|
sl@0
|
190 |
TCell represents the structure of header cells.
|
sl@0
|
191 |
Each cells contains the following fields:
|
sl@0
|
192 |
- Flat buffer field type;
|
sl@0
|
193 |
- Flat buffer field length;
|
sl@0
|
194 |
- Flat buffer field data position;
|
sl@0
|
195 |
= if 0 then the field is "Not present" (no memory is reserved for the field data, the field data is missing);
|
sl@0
|
196 |
= if positive - the field data position in the flat buffer (counting from the beginning of the header);
|
sl@0
|
197 |
= if negative - the field is "Not present", but memory has been reserved for it;
|
sl@0
|
198 |
|
sl@0
|
199 |
RSqlBufFlat class offers fast, indexed access to the header cells.
|
sl@0
|
200 |
|
sl@0
|
201 |
@see RSqlBufFlat
|
sl@0
|
202 |
*/
|
sl@0
|
203 |
struct TCell
|
sl@0
|
204 |
{
|
sl@0
|
205 |
inline TInt Type() const {return (iBits >> RSqlBufFlat::EWidthLen) & (RSqlBufFlat::EMaxType - 1);}
|
sl@0
|
206 |
inline TInt Size() const {return (iBits & (RSqlBufFlat::EMaxLength - 1));}
|
sl@0
|
207 |
TUint32 iBits; //element type: EWidthType bits & element length: EWidthLen bits.
|
sl@0
|
208 |
TInt32 iPos; //element data position, relative to the beginning of the header.
|
sl@0
|
209 |
// Zero if the element is not present.
|
sl@0
|
210 |
// Negative if the element is not present but memory is reserved.
|
sl@0
|
211 |
};
|
sl@0
|
212 |
|
sl@0
|
213 |
enum {EExpandSize = 256}; //iBuf min expansion size - it must be a power of 2 and
|
sl@0
|
214 |
// (EExpandSize % 8) should be 0.
|
sl@0
|
215 |
/**
|
sl@0
|
216 |
TBufFlat structure represents the "SysData" part of the flat buffer (see RSqlBufFlat comments), beyond which
|
sl@0
|
217 |
begins the flat buffer header and "Data" part.
|
sl@0
|
218 |
|
sl@0
|
219 |
TBufFlat structure contains information about:
|
sl@0
|
220 |
- Flat buffer elements count;
|
sl@0
|
221 |
- Flat buffer header size;
|
sl@0
|
222 |
- The size of used part of the flat buffer;
|
sl@0
|
223 |
|
sl@0
|
224 |
@see RSqlBufFlat
|
sl@0
|
225 |
*/
|
sl@0
|
226 |
struct TBufFlat
|
sl@0
|
227 |
{
|
sl@0
|
228 |
TInt iCount; //element count
|
sl@0
|
229 |
TInt iHeaderSize; //buffer header size
|
sl@0
|
230 |
TInt iSize; //used buffer size
|
sl@0
|
231 |
TInt iReserved; //maintains TBufFlat 8-byte aligned
|
sl@0
|
232 |
};
|
sl@0
|
233 |
|
sl@0
|
234 |
private: //Method declarations
|
sl@0
|
235 |
void DoInit();
|
sl@0
|
236 |
TInt DoReAlloc(TInt aSize);
|
sl@0
|
237 |
inline TInt Available() const;
|
sl@0
|
238 |
void DoSet(TInt aIndex, TInt aType, const void* aData, TInt aDataLength);
|
sl@0
|
239 |
TInt Reserve(TInt aLength);
|
sl@0
|
240 |
void Invariant() const;
|
sl@0
|
241 |
inline const TCell* Header() const;
|
sl@0
|
242 |
inline TCell* Header();
|
sl@0
|
243 |
inline TInt SysDataSize() const;
|
sl@0
|
244 |
|
sl@0
|
245 |
private: //Data declarations
|
sl@0
|
246 |
// IPC data
|
sl@0
|
247 |
TBufFlat* iBuf;
|
sl@0
|
248 |
// non-IPC data
|
sl@0
|
249 |
TInt iMaxSize; //max buffer size (allocated memory), not part of RSqlBufFlat data (IPC - sent/received).
|
sl@0
|
250 |
//iMaxSize can't be part of the IPC data, because it may be overwritten, which leads to
|
sl@0
|
251 |
//"hard to detect" problems. For example:
|
sl@0
|
252 |
// - client sends to the server in/out flat buffer with max size 256 bytes (allocated);
|
sl@0
|
253 |
// - the server has a flat buffer which max size is 512, but the used size is 200;
|
sl@0
|
254 |
// - the client flat buffer has enough space for those 200 bytes, so the server will
|
sl@0
|
255 |
// copy its flat buffer content to the client flat buffer. The client's flat buffer
|
sl@0
|
256 |
// iMaxSize data member will lose its original value (256) and will have new value assigned -
|
sl@0
|
257 |
// 512. But the client flat buffer does not have 512 bytes allocated!
|
sl@0
|
258 |
mutable TPtrC8 iBufPtrC; //it is set to point to iBuf. Not part of RSqlBufFlat data (IPC - sent/received)
|
sl@0
|
259 |
mutable TPtr8 iBufPtr; //it is set to point to iBuf. Not part of RSqlBufFlat data (IPC - sent/received)
|
sl@0
|
260 |
};
|
sl@0
|
261 |
|
sl@0
|
262 |
#include "SqlBufFlat.inl"
|
sl@0
|
263 |
|
sl@0
|
264 |
#endif//__SQLBUFFLAT_H__
|