1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*************************************************************************
4 * The Contents of this file are made available subject to the terms of
5 * either of the following licenses
7 * - GNU Lesser General Public License Version 2.1
8 * - Sun Industry Standards Source License Version 1.1
10 * Sun Microsystems Inc., October, 2000
12 * GNU Lesser General Public License Version 2.1
13 * =============================================
14 * Copyright 2000 by Sun Microsystems, Inc.
15 * 901 San Antonio Road, Palo Alto, CA 94303, USA
17 * This library is free software; you can redistribute it and/or
18 * modify it under the terms of the GNU Lesser General Public
19 * License version 2.1, as published by the Free Software Foundation.
21 * This library is distributed in the hope that it will be useful,
22 * but WITHOUT ANY WARRANTY; without even the implied warranty of
23 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
24 * Lesser General Public License for more details.
26 * You should have received a copy of the GNU Lesser General Public
27 * License along with this library; if not, write to the Free Software
28 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
32 * Sun Industry Standards Source License Version 1.1
33 * =================================================
34 * The contents of this file are subject to the Sun Industry Standards
35 * Source License Version 1.1 (the "License"); You may not use this file
36 * except in compliance with the License. You may obtain a copy of the
37 * License at http://www.openoffice.org/license.html.
39 * Software provided under this License is provided on an "AS IS" basis,
40 * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
41 * WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
42 * MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
43 * See the License for the specific provisions governing your rights and
44 * obligations concerning the Software.
46 * The Initial Developer of the Original Code is: IBM Corporation
48 * Copyright: 2008 by IBM Corporation
50 * All Rights Reserved.
52 * Contributor(s): _______________________________________
55 ************************************************************************/
58 namespace OpenStormBento
62 CBenTOCReader::ReadLabelAndTOC()
66 unsigned long TOCOffset
;
67 if ((Err
= ReadLabel(&TOCOffset
, &cTOCSize
)) != BenErr_OK
)
70 unsigned long nLength
;
71 if ((Err
= cpContainer
->GetSize(&nLength
)) != BenErr_OK
)
74 if (TOCOffset
> nLength
)
75 return BenErr_ReadPastEndOfTOC
;
77 if (cTOCSize
> nLength
- TOCOffset
)
78 return BenErr_ReadPastEndOfTOC
;
80 if ((Err
= cpContainer
->SeekToPosition(TOCOffset
)) != BenErr_OK
)
83 cpTOC
= new BenByte
[cTOCSize
];
84 if ((Err
= cpContainer
->ReadKnownSize(cpTOC
, cTOCSize
)) != BenErr_OK
)
87 if ((Err
= ReadTOC()) != BenErr_OK
)
94 CBenTOCReader::ReadLabel(unsigned long * pTOCOffset
, unsigned long * pTOCSize
)
96 // If seek fails, then probably because stream is smaller than
97 // BEN_LABEL_SIZE and thus can't be Bento container
99 if ((Err
= cpContainer
->SeekFromEnd(-BEN_LABEL_SIZE
)) != BenErr_OK
)
100 return BenErr_NotBentoContainer
;
102 BenByte Label
[BEN_LABEL_SIZE
];
103 if ((Err
= cpContainer
->ReadKnownSize(Label
, BEN_LABEL_SIZE
)) != BenErr_OK
)
106 if (memcmp(Label
, gsBenMagicBytes
, BEN_MAGIC_BYTES_SIZE
) != 0)
107 if ((Err
= SearchForLabel(Label
)) != BenErr_OK
)
110 BenByte
* pCurrLabel
= Label
+ BEN_MAGIC_BYTES_SIZE
;
115 UtGetIntelWord(pCurrLabel
); pCurrLabel
+= 2; // Flags
116 // Newer files are 0x0101--indicates if big or little endian. Older
117 // files are 0x0 for flags
118 assert(Flags
== 0x0101 || Flags
== 0x0);
120 cBlockSize
= UtGetIntelWord(pCurrLabel
) * 1024; pCurrLabel
+= 2;
122 // Check major version
123 if (UtGetIntelWord(pCurrLabel
) != BEN_CURR_MAJOR_VERSION
)
124 return BenErr_UnknownBentoFormatVersion
;
127 UtGetIntelWord(pCurrLabel
); pCurrLabel
+= 2; // Minor version
129 *pTOCOffset
= UtGetIntelDWord(pCurrLabel
); pCurrLabel
+= 4;
130 *pTOCSize
= UtGetIntelDWord(pCurrLabel
);
132 assert(pCurrLabel
+ 4 == Label
+ BEN_LABEL_SIZE
);
137 #define LABEL_READ_BUFFER_SIZE 500
138 #define MAX_SEARCH_AMOUNT 1024 * 1024
141 CBenTOCReader::SearchForLabel(BenByte
* pLabel
)
146 if ((Err
= cpContainer
->GetSize(&Length
)) != BenErr_OK
)
149 // Always ready to check for MagicBytes from
150 // CurrOffset - BEN_MAGIC_BYTES_SIZE to CurrOffset - 1
151 unsigned long CurrOffset
= Length
- BEN_LABEL_SIZE
+ BEN_MAGIC_BYTES_SIZE
-
154 char Buffer
[LABEL_READ_BUFFER_SIZE
] = {0};
156 unsigned long BufferStartOffset
= Length
; // Init to big value
158 while (CurrOffset
>= BEN_MAGIC_BYTES_SIZE
)
160 // Don't search backwards more than 1 meg
161 if (Length
- CurrOffset
> MAX_SEARCH_AMOUNT
)
164 // If before beginning of buffer
165 if (CurrOffset
- BEN_MAGIC_BYTES_SIZE
< BufferStartOffset
)
167 unsigned long UsedBufferSize
;
168 if (CurrOffset
< LABEL_READ_BUFFER_SIZE
)
169 UsedBufferSize
= CurrOffset
;
170 else UsedBufferSize
= LABEL_READ_BUFFER_SIZE
;
172 if ((Err
= cpContainer
->SeekToPosition(CurrOffset
- UsedBufferSize
))
176 if ((Err
= cpContainer
->ReadKnownSize(Buffer
, UsedBufferSize
)) !=
180 BufferStartOffset
= CurrOffset
- UsedBufferSize
;
183 if (memcmp(Buffer
+ (CurrOffset
- BEN_MAGIC_BYTES_SIZE
-
184 BufferStartOffset
), gsBenMagicBytes
, BEN_MAGIC_BYTES_SIZE
) == 0)
186 if ((Err
= cpContainer
->SeekToPosition(CurrOffset
-
187 BEN_MAGIC_BYTES_SIZE
)) != BenErr_OK
)
190 return cpContainer
->ReadKnownSize(pLabel
, BEN_LABEL_SIZE
);
196 return BenErr_NotBentoContainer
; // Didn't find magic bytes
200 CBenTOCReader::ReadTOC()
203 BenByte LookAhead
= GetCode();
204 BenGeneration Generation
= 0;
206 // Read in all objects
207 while (LookAhead
== BEN_NEW_OBJECT
)
209 BenObjectID ObjectID
;
210 if ((Err
= GetDWord(&ObjectID
)) != BenErr_OK
)
212 pCBenObject pObject
= NULL
;
214 // Read in all properties for object
217 BenObjectID PropertyID
;
219 if ((Err
= GetDWord(&PropertyID
)) != BenErr_OK
)
221 pCBenProperty pProperty
= NULL
;
223 // Read in all values for property
226 BenObjectID ReferencedListID
= 0;
229 if ((Err
= GetDWord(&TypeID
)) != BenErr_OK
)
231 LookAhead
= GetCode();
233 if (LookAhead
== BEN_EXPLICIT_GEN
)
235 if ((Err
= GetDWord(&Generation
)) != BenErr_OK
)
237 LookAhead
= GetCode();
240 if (LookAhead
== BEN_REFERENCE_LIST_ID
)
242 if ((Err
= GetDWord(&ReferencedListID
)) != BenErr_OK
)
244 LookAhead
= GetCode();
247 if (PropertyID
== BEN_PROPID_GLOBAL_PROPERTY_NAME
||
248 PropertyID
== BEN_PROPID_GLOBAL_TYPE_NAME
)
250 // Read property or type name
252 if (pObject
!= NULL
|| TypeID
!= BEN_TYPEID_7_BIT_ASCII
||
253 LookAhead
!= BEN_OFFSET4_LEN4
)
254 return BenErr_NamedObjectError
;
257 unsigned long Length
;
259 if ((Err
= GetDWord(&Pos
)) != BenErr_OK
)
261 if ((Err
= GetDWord(&Length
)) != BenErr_OK
)
263 LookAhead
= GetCode();
265 if ((Err
= cpContainer
->SeekToPosition(Pos
)) != BenErr_OK
)
268 #define STACK_BUFFER_SIZE 256
269 char sStackBuffer
[STACK_BUFFER_SIZE
];
272 if (Length
> STACK_BUFFER_SIZE
)
274 sBuffer
= new char[Length
];
275 sAllocBuffer
= sBuffer
;
279 sBuffer
= sStackBuffer
;
283 if ((Err
= cpContainer
->ReadKnownSize(sBuffer
, Length
)) !=
286 delete[] sAllocBuffer
;
290 pCBenNamedObjectListElmt pPrevNamedObjectListElmt
;
291 if (FindNamedObject(cpContainer
->GetNamedObjects(),
292 sBuffer
, &pPrevNamedObjectListElmt
) != NULL
)
294 delete[] sAllocBuffer
;
295 return BenErr_DuplicateName
;
298 pCBenObject pPrevObject
= (pCBenObject
) cpContainer
->
299 GetObjects()->GetLast();
301 if (PropertyID
== BEN_PROPID_GLOBAL_PROPERTY_NAME
)
302 pObject
= new CBenPropertyName(cpContainer
, ObjectID
,
303 pPrevObject
, sBuffer
, pPrevNamedObjectListElmt
);
304 else pObject
= new CBenTypeName(cpContainer
, ObjectID
,
305 pPrevObject
, sBuffer
, pPrevNamedObjectListElmt
);
307 delete[] sAllocBuffer
;
309 else if (PropertyID
== BEN_PROPID_OBJ_REFERENCES
)
311 // Don't need to read in references object--we assume
312 // that all references use object ID as key
313 if ((Err
= ReadSegments(NULL
, &LookAhead
)) != BenErr_OK
)
316 else if (ObjectID
== BEN_OBJID_TOC
)
318 if (PropertyID
== BEN_PROPID_TOC_SEED
)
320 if (TypeID
!= BEN_TYPEID_TOC_TYPE
||
321 LookAhead
!= BEN_IMMEDIATE4
)
322 return BenErr_TOCSeedError
;
325 if ((Err
= GetDWord(&Data
)) != BenErr_OK
)
327 LookAhead
= GetCode();
329 cpContainer
->SetNextAvailObjectID(Data
);
333 // Ignore the other BEN_OBJID_TOC properties
334 if ((Err
= ReadSegments(NULL
, &LookAhead
)) != BenErr_OK
)
340 if (pProperty
!= NULL
)
341 return BenErr_PropertyWithMoreThanOneValue
;
344 pObject
= new CBenObject(cpContainer
, ObjectID
,
345 (pCBenObject
) cpContainer
->GetObjects()->GetLast());
347 pProperty
= new CBenProperty(pObject
, PropertyID
, TypeID
,
348 (pCBenProperty
) pObject
->GetProperties()->GetLast());
350 if ((Err
= ReadSegments(pProperty
->UseValue(),
351 &LookAhead
)) != BenErr_OK
)
354 } while (LookAhead
== BEN_NEW_TYPE
);
355 } while (LookAhead
== BEN_NEW_PROPERTY
);
358 if (LookAhead
== BEN_READ_PAST_END_OF_TOC
)
360 else return BenErr_InvalidTOC
;
364 CBenTOCReader::ReadSegments(pCBenValue pValue
, BenByte
* pLookAhead
)
368 while (*pLookAhead
>= BEN_SEGMENT_CODE_START
&&
369 *pLookAhead
<= BEN_SEGMENT_CODE_END
)
371 if ((Err
= ReadSegment(pValue
, pLookAhead
)) !=
380 CBenTOCReader::ReadSegment(pCBenValue pValue
, BenByte
* pLookAhead
)
384 bool Immediate
= false;
385 bool EightByteOffset
= false;
386 unsigned long Offset(0), Length(0);
390 case BEN_CONT_OFFSET4_LEN4
:
391 case BEN_OFFSET4_LEN4
:
392 if ((Err
= GetDWord(&Offset
)) != BenErr_OK
)
394 if ((Err
= GetDWord(&Length
)) != BenErr_OK
)
399 Length
= 0; Immediate
= true;
403 Length
= 1; Immediate
= true;
407 Length
= 2; Immediate
= true;
411 Length
= 3; Immediate
= true;
414 case BEN_CONT_IMMEDIATE4
:
416 Length
= 4; Immediate
= true;
419 case BEN_CONT_OFFSET8_LEN4
:
420 case BEN_OFFSET8_LEN4
:
421 EightByteOffset
= true;
429 if (Immediate
&& Length
!= 0)
430 if ((Err
= GetData(ImmData
, 4)) != BenErr_OK
)
433 *pLookAhead
= GetCode();
436 return BenErr_64BitOffsetNotSupported
;
441 new CBenValueSegment(pValue
, Offset
, Length
);
442 else if (Length
!= 0)
445 new CBenValueSegment(pValue
, ImmData
, (unsigned short) Length
);
453 CBenTOCReader::CanGetData(unsigned long Amt
)
455 return cCurr
+ Amt
<= cTOCSize
;
459 CBenTOCReader::GetByte(BenByte
* pByte
)
462 return BenErr_ReadPastEndOfTOC
;
464 *pByte
= UtGetIntelByte(cpTOC
+ cCurr
);
470 CBenTOCReader::GetDWord(BenDWord
* pDWord
)
473 return BenErr_ReadPastEndOfTOC
;
475 *pDWord
= UtGetIntelDWord(cpTOC
+ cCurr
);
481 CBenTOCReader::GetCode()
486 if (GetByte(&Code
) != BenErr_OK
)
487 return BEN_READ_PAST_END_OF_TOC
;
489 if (Code
== BEN_END_OF_BUFFER
)
490 // Advance to next block
491 cCurr
= cBlockSize
* ((cCurr
+ (cBlockSize
- 1)) /
494 while (Code
== BEN_NOOP
|| Code
== BEN_END_OF_BUFFER
);
499 CBenTOCReader::GetData(BenDataPtr pBuffer
, unsigned long Amt
)
501 if (! CanGetData(Amt
))
502 return BenErr_ReadPastEndOfTOC
;
504 UtHugeMemcpy(pBuffer
, cpTOC
+ cCurr
, Amt
);
508 }//end OpenStormBento namespace
510 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */