xref: /plan9/sys/src/cmd/aux/antiword/prop6.c (revision 25b329d522281a8cdd35da0dcc08c3fc621059a9)
1f5736e95SDavid du Colombier /*
2f5736e95SDavid du Colombier  * prop6.c
3*25b329d5SDavid du Colombier  * Copyright (C) 1998-2005 A.J. van Os; Released under GPL
4f5736e95SDavid du Colombier  *
5f5736e95SDavid du Colombier  * Description:
6f5736e95SDavid du Colombier  * Read the property information from a MS Word 6 or 7 file
7f5736e95SDavid du Colombier  */
8f5736e95SDavid du Colombier 
9f5736e95SDavid du Colombier #include <stdlib.h>
10f5736e95SDavid du Colombier #include <string.h>
11f5736e95SDavid du Colombier #include "antiword.h"
12f5736e95SDavid du Colombier 
13f5736e95SDavid du Colombier 
14f5736e95SDavid du Colombier /*
15f5736e95SDavid du Colombier  * iGet6InfoLength - the length of the information for Word 6/7 files
16f5736e95SDavid du Colombier  */
17f5736e95SDavid du Colombier static int
iGet6InfoLength(int iByteNbr,const UCHAR * aucGrpprl)18f5736e95SDavid du Colombier iGet6InfoLength(int iByteNbr, const UCHAR *aucGrpprl)
19f5736e95SDavid du Colombier {
20f5736e95SDavid du Colombier 	int	iTmp, iDel, iAdd;
21f5736e95SDavid du Colombier 
22f5736e95SDavid du Colombier 	switch (ucGetByte(iByteNbr, aucGrpprl)) {
23f5736e95SDavid du Colombier 	case   2: case  16: case  17: case  18: case  19: case  21: case  22:
24f5736e95SDavid du Colombier 	case  26: case  27: case  28: case  30: case  31: case  32: case  33:
25f5736e95SDavid du Colombier 	case  34: case  35: case  36: case  38: case  39: case  40: case  41:
26f5736e95SDavid du Colombier 	case  42: case  43: case  45: case  46: case  47: case  48: case  49:
27f5736e95SDavid du Colombier 	case  69: case  72: case  80: case  93: case  96: case  97: case  99:
28f5736e95SDavid du Colombier 	case 101: case 105: case 106: case 107: case 109: case 110: case 121:
29f5736e95SDavid du Colombier 	case 122: case 123: case 124: case 140: case 141: case 144: case 145:
30f5736e95SDavid du Colombier 	case 148: case 149: case 154: case 155: case 156: case 157: case 160:
31f5736e95SDavid du Colombier 	case 161: case 164: case 165: case 166: case 167: case 168: case 169:
32f5736e95SDavid du Colombier 	case 170: case 171: case 182: case 183: case 184: case 189: case 195:
33f5736e95SDavid du Colombier 	case 197: case 198:
34f5736e95SDavid du Colombier 		return 1 + 2;
35f5736e95SDavid du Colombier 	case   3: case  12: case  15: case  81: case 103: case 108: case 188:
36f5736e95SDavid du Colombier 	case 190: case 191:
37f5736e95SDavid du Colombier 		return 2 + (int)ucGetByte(iByteNbr + 1, aucGrpprl);
38f5736e95SDavid du Colombier 	case  20: case  70: case  74: case 192: case 194: case 196: case 200:
39f5736e95SDavid du Colombier 		return 1 + 4;
40f5736e95SDavid du Colombier 	case  23:
41f5736e95SDavid du Colombier 		iTmp = (int)ucGetByte(iByteNbr + 1, aucGrpprl);
42f5736e95SDavid du Colombier 		if (iTmp == 255) {
43f5736e95SDavid du Colombier 			iDel = (int)ucGetByte(iByteNbr + 2, aucGrpprl);
44f5736e95SDavid du Colombier 			iAdd = (int)ucGetByte(
45f5736e95SDavid du Colombier 					iByteNbr + 3 + iDel * 4, aucGrpprl);
46f5736e95SDavid du Colombier 			iTmp = 2 + iDel * 4 + iAdd * 3;
47f5736e95SDavid du Colombier 		}
48f5736e95SDavid du Colombier 		return 2 + iTmp;
49f5736e95SDavid du Colombier 	case  68: case 193: case 199:
50f5736e95SDavid du Colombier 		return 1 + 5;
51f5736e95SDavid du Colombier 	case  73: case  95: case 136: case 137:
52f5736e95SDavid du Colombier 		return 1 + 3;
53*25b329d5SDavid du Colombier 	case 120: case 187:
54f5736e95SDavid du Colombier 		return 1 + 12;
55f5736e95SDavid du Colombier 	default:
56f5736e95SDavid du Colombier 		return 1 + 1;
57f5736e95SDavid du Colombier 	}
58f5736e95SDavid du Colombier } /* end of iGet6InfoLength */
59f5736e95SDavid du Colombier 
60f5736e95SDavid du Colombier /*
61*25b329d5SDavid du Colombier  * Build the lists with Document Property Information for Word 6/7 files
62*25b329d5SDavid du Colombier  */
63*25b329d5SDavid du Colombier void
vGet6DopInfo(FILE * pFile,ULONG ulStartBlock,const ULONG * aulBBD,size_t tBBDLen,const UCHAR * aucHeader)64*25b329d5SDavid du Colombier vGet6DopInfo(FILE *pFile, ULONG ulStartBlock,
65*25b329d5SDavid du Colombier 	const ULONG *aulBBD, size_t tBBDLen,
66*25b329d5SDavid du Colombier 	const UCHAR *aucHeader)
67*25b329d5SDavid du Colombier {
68*25b329d5SDavid du Colombier 	document_block_type	tDocument;
69*25b329d5SDavid du Colombier 	UCHAR	*aucBuffer;
70*25b329d5SDavid du Colombier 	ULONG	ulBeginDocpInfo, ulTmp;
71*25b329d5SDavid du Colombier 	size_t	tDocpInfoLen;
72*25b329d5SDavid du Colombier 	USHORT	usTmp;
73*25b329d5SDavid du Colombier 
74*25b329d5SDavid du Colombier 	ulBeginDocpInfo = ulGetLong(0x150, aucHeader); /* fcDop */
75*25b329d5SDavid du Colombier 	DBG_HEX(ulBeginDocpInfo);
76*25b329d5SDavid du Colombier 	tDocpInfoLen = (size_t)ulGetLong(0x154, aucHeader); /* lcbDop */
77*25b329d5SDavid du Colombier 	DBG_DEC(tDocpInfoLen);
78*25b329d5SDavid du Colombier 	if (tDocpInfoLen < 28) {
79*25b329d5SDavid du Colombier 		DBG_MSG("No Document information");
80*25b329d5SDavid du Colombier 		return;
81*25b329d5SDavid du Colombier 	}
82*25b329d5SDavid du Colombier 
83*25b329d5SDavid du Colombier 	aucBuffer = xmalloc(tDocpInfoLen);
84*25b329d5SDavid du Colombier 	if (!bReadBuffer(pFile, ulStartBlock,
85*25b329d5SDavid du Colombier 			aulBBD, tBBDLen, BIG_BLOCK_SIZE,
86*25b329d5SDavid du Colombier 			aucBuffer, ulBeginDocpInfo, tDocpInfoLen)) {
87*25b329d5SDavid du Colombier 		aucBuffer = xfree(aucBuffer);
88*25b329d5SDavid du Colombier 		return;
89*25b329d5SDavid du Colombier 	}
90*25b329d5SDavid du Colombier 
91*25b329d5SDavid du Colombier 	usTmp = usGetWord(0x00, aucBuffer);
92*25b329d5SDavid du Colombier 	tDocument.ucHdrFtrSpecification = (UCHAR)(usTmp >> 8); /* grpfIhdt */
93*25b329d5SDavid du Colombier 	tDocument.usDefaultTabWidth = usGetWord(0x0a, aucBuffer); /* dxaTab */
94*25b329d5SDavid du Colombier 	ulTmp = ulGetLong(0x14, aucBuffer); /* dttmCreated */
95*25b329d5SDavid du Colombier 	tDocument.tCreateDate = tConvertDTTM(ulTmp);
96*25b329d5SDavid du Colombier 	ulTmp = ulGetLong(0x18, aucBuffer); /* dttmRevised */
97*25b329d5SDavid du Colombier 	tDocument.tRevisedDate = tConvertDTTM(ulTmp);
98*25b329d5SDavid du Colombier 	vCreateDocumentInfoList(&tDocument);
99*25b329d5SDavid du Colombier 
100*25b329d5SDavid du Colombier 	aucBuffer = xfree(aucBuffer);
101*25b329d5SDavid du Colombier } /* end of vGet6DopInfo */
102*25b329d5SDavid du Colombier 
103*25b329d5SDavid du Colombier /*
104f5736e95SDavid du Colombier  * Fill the section information block with information
105f5736e95SDavid du Colombier  * from a Word 6/7 file.
106f5736e95SDavid du Colombier  */
107f5736e95SDavid du Colombier static void
vGet6SectionInfo(const UCHAR * aucGrpprl,size_t tBytes,section_block_type * pSection)108f5736e95SDavid du Colombier vGet6SectionInfo(const UCHAR *aucGrpprl, size_t tBytes,
109f5736e95SDavid du Colombier 		section_block_type *pSection)
110f5736e95SDavid du Colombier {
111f5736e95SDavid du Colombier 	UINT	uiIndex;
112f5736e95SDavid du Colombier 	int	iFodoOff, iInfoLen, iSize, iTmp;
113f5736e95SDavid du Colombier 	USHORT	usCcol;
114f5736e95SDavid du Colombier 	UCHAR	ucTmp;
115f5736e95SDavid du Colombier 
116f5736e95SDavid du Colombier 	fail(aucGrpprl == NULL || pSection == NULL);
117f5736e95SDavid du Colombier 
118f5736e95SDavid du Colombier 	iFodoOff = 0;
119f5736e95SDavid du Colombier 	while (tBytes >= (size_t)iFodoOff + 1) {
120f5736e95SDavid du Colombier 		iInfoLen = 0;
121f5736e95SDavid du Colombier 		switch (ucGetByte(iFodoOff, aucGrpprl)) {
122f5736e95SDavid du Colombier 		case 133:	/* olstAnm */
123f5736e95SDavid du Colombier 			iSize = (int)ucGetByte(iFodoOff + 1, aucGrpprl);
124f5736e95SDavid du Colombier 			DBG_DEC_C(iSize != 212, iSize);
125f5736e95SDavid du Colombier 			for (uiIndex = 0, iTmp = iFodoOff + 2;
126f5736e95SDavid du Colombier 			     uiIndex < 9 && iTmp < iFodoOff + 2 + iSize - 15;
127f5736e95SDavid du Colombier 			     uiIndex++, iTmp += 16) {
128f5736e95SDavid du Colombier 				pSection->aucNFC[uiIndex] =
129f5736e95SDavid du Colombier 						ucGetByte(iTmp, aucGrpprl);
130f5736e95SDavid du Colombier 				NO_DBG_DEC(pSection->aucNFC[uiIndex]);
131f5736e95SDavid du Colombier 				ucTmp = ucGetByte(iTmp + 3, aucGrpprl);
132f5736e95SDavid du Colombier 				NO_DBG_HEX(ucTmp);
133f5736e95SDavid du Colombier 				if ((ucTmp & BIT(2)) != 0) {
134f5736e95SDavid du Colombier 					pSection->usNeedPrevLvl |=
135f5736e95SDavid du Colombier 							(USHORT)BIT(uiIndex);
136f5736e95SDavid du Colombier 				}
137f5736e95SDavid du Colombier 				if ((ucTmp & BIT(3)) != 0) {
138f5736e95SDavid du Colombier 					pSection->usHangingIndent |=
139f5736e95SDavid du Colombier 							(USHORT)BIT(uiIndex);
140f5736e95SDavid du Colombier 				}
141f5736e95SDavid du Colombier 			}
142f5736e95SDavid du Colombier 			DBG_HEX(pSection->usNeedPrevLvl);
143f5736e95SDavid du Colombier 			DBG_HEX(pSection->usHangingIndent);
144f5736e95SDavid du Colombier 			break;
145f5736e95SDavid du Colombier 		case 142:	/* bkc */
146f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodoOff + 1, aucGrpprl);
147f5736e95SDavid du Colombier 			DBG_DEC(ucTmp);
148f5736e95SDavid du Colombier 			pSection->bNewPage = ucTmp != 0 && ucTmp != 1;
149f5736e95SDavid du Colombier 			break;
150f5736e95SDavid du Colombier 		case 144:	/* ccolM1 */
151f5736e95SDavid du Colombier 			usCcol = 1 + usGetWord(iFodoOff + 1, aucGrpprl);
152f5736e95SDavid du Colombier 			DBG_DEC(usCcol);
153f5736e95SDavid du Colombier 			break;
154*25b329d5SDavid du Colombier 		case 153:	/* grpfIhdt */
155*25b329d5SDavid du Colombier 			pSection->ucHdrFtrSpecification =
156*25b329d5SDavid du Colombier 					ucGetByte(iFodoOff + 1, aucGrpprl);
157*25b329d5SDavid du Colombier 			break;
158f5736e95SDavid du Colombier 		default:
159f5736e95SDavid du Colombier 			break;
160f5736e95SDavid du Colombier 		}
161f5736e95SDavid du Colombier 		if (iInfoLen <= 0) {
162f5736e95SDavid du Colombier 			iInfoLen = iGet6InfoLength(iFodoOff, aucGrpprl);
163f5736e95SDavid du Colombier 			fail(iInfoLen <= 0);
164f5736e95SDavid du Colombier 		}
165f5736e95SDavid du Colombier 		iFodoOff += iInfoLen;
166f5736e95SDavid du Colombier 	}
167f5736e95SDavid du Colombier } /* end of vGet6SectionInfo */
168f5736e95SDavid du Colombier 
169f5736e95SDavid du Colombier /*
170f5736e95SDavid du Colombier  * Build the lists with Section Property Information for Word 6/7 files
171f5736e95SDavid du Colombier  */
172f5736e95SDavid du Colombier void
vGet6SepInfo(FILE * pFile,ULONG ulStartBlock,const ULONG * aulBBD,size_t tBBDLen,const UCHAR * aucHeader)173f5736e95SDavid du Colombier vGet6SepInfo(FILE *pFile, ULONG ulStartBlock,
174f5736e95SDavid du Colombier 	const ULONG *aulBBD, size_t tBBDLen,
175f5736e95SDavid du Colombier 	const UCHAR *aucHeader)
176f5736e95SDavid du Colombier {
177f5736e95SDavid du Colombier 	section_block_type	tSection;
178*25b329d5SDavid du Colombier 	ULONG		*aulSectPage, *aulCharPos;
179f5736e95SDavid du Colombier 	UCHAR	*aucBuffer, *aucFpage;
180*25b329d5SDavid du Colombier 	ULONG	ulBeginOfText, ulTextOffset, ulBeginSectInfo;
181*25b329d5SDavid du Colombier 	size_t	tSectInfoLen, tIndex, tOffset, tLen, tBytes;
182f5736e95SDavid du Colombier 	UCHAR	aucTmp[2];
183f5736e95SDavid du Colombier 
184f5736e95SDavid du Colombier 	fail(pFile == NULL || aucHeader == NULL);
185f5736e95SDavid du Colombier 	fail(ulStartBlock > MAX_BLOCKNUMBER && ulStartBlock != END_OF_CHAIN);
186f5736e95SDavid du Colombier 	fail(aulBBD == NULL);
187f5736e95SDavid du Colombier 
188*25b329d5SDavid du Colombier         ulBeginOfText = ulGetLong(0x18, aucHeader); /* fcMin */
189*25b329d5SDavid du Colombier         NO_DBG_HEX(ulBeginOfText);
190f5736e95SDavid du Colombier 	ulBeginSectInfo = ulGetLong(0x88, aucHeader); /* fcPlcfsed */
191f5736e95SDavid du Colombier 	DBG_HEX(ulBeginSectInfo);
192f5736e95SDavid du Colombier 	tSectInfoLen = (size_t)ulGetLong(0x8c, aucHeader); /* lcbPlcfsed */
193f5736e95SDavid du Colombier 	DBG_DEC(tSectInfoLen);
194f5736e95SDavid du Colombier 	if (tSectInfoLen < 4) {
195f5736e95SDavid du Colombier 		DBG_DEC(tSectInfoLen);
196f5736e95SDavid du Colombier 		return;
197f5736e95SDavid du Colombier 	}
198f5736e95SDavid du Colombier 
199f5736e95SDavid du Colombier 	aucBuffer = xmalloc(tSectInfoLen);
200f5736e95SDavid du Colombier 	if (!bReadBuffer(pFile, ulStartBlock,
201f5736e95SDavid du Colombier 			aulBBD, tBBDLen, BIG_BLOCK_SIZE,
202f5736e95SDavid du Colombier 			aucBuffer, ulBeginSectInfo, tSectInfoLen)) {
203f5736e95SDavid du Colombier 		aucBuffer = xfree(aucBuffer);
204f5736e95SDavid du Colombier 		return;
205f5736e95SDavid du Colombier 	}
206f5736e95SDavid du Colombier 	NO_DBG_PRINT_BLOCK(aucBuffer, tSectInfoLen);
207f5736e95SDavid du Colombier 
208f5736e95SDavid du Colombier 	/* Read the Section Descriptors */
209f5736e95SDavid du Colombier 	tLen = (tSectInfoLen - 4) / 16;
210f5736e95SDavid du Colombier 	/* Save the section offsets */
211*25b329d5SDavid du Colombier 	aulCharPos = xcalloc(tLen, sizeof(ULONG));
212*25b329d5SDavid du Colombier 	for (tIndex = 0, tOffset = 0; tIndex < tLen; tIndex++, tOffset += 4) {
213*25b329d5SDavid du Colombier 		ulTextOffset = ulGetLong(tOffset, aucBuffer);
214*25b329d5SDavid du Colombier 		NO_DBG_HEX(ulTextOffset);
215*25b329d5SDavid du Colombier 		aulCharPos[tIndex] = ulBeginOfText + ulTextOffset;
216*25b329d5SDavid du Colombier 		NO_DBG_HEX(aulCharPos[tIndex]);
217f5736e95SDavid du Colombier 	}
218f5736e95SDavid du Colombier 	/* Save the Sepx offsets */
219f5736e95SDavid du Colombier 	aulSectPage = xcalloc(tLen, sizeof(ULONG));
220*25b329d5SDavid du Colombier 	for (tIndex = 0, tOffset = (tLen + 1) * 4;
221*25b329d5SDavid du Colombier 	     tIndex < tLen;
222*25b329d5SDavid du Colombier 	     tIndex++, tOffset += 12) {
223*25b329d5SDavid du Colombier 		aulSectPage[tIndex] = ulGetLong(tOffset + 2, aucBuffer);
224*25b329d5SDavid du Colombier 		NO_DBG_HEX(aulSectPage[tIndex]); /* fcSepx */
225f5736e95SDavid du Colombier 	}
226f5736e95SDavid du Colombier 	aucBuffer = xfree(aucBuffer);
227f5736e95SDavid du Colombier 
228f5736e95SDavid du Colombier 	/* Read the Section Properties */
229*25b329d5SDavid du Colombier 	for (tIndex = 0; tIndex < tLen; tIndex++) {
230*25b329d5SDavid du Colombier 		if (aulSectPage[tIndex] == FC_INVALID) {
231*25b329d5SDavid du Colombier 			vDefault2SectionInfoList(aulCharPos[tIndex]);
232f5736e95SDavid du Colombier 			continue;
233f5736e95SDavid du Colombier 		}
234f5736e95SDavid du Colombier 		/* Get the number of bytes to read */
235f5736e95SDavid du Colombier 		if (!bReadBuffer(pFile, ulStartBlock,
236f5736e95SDavid du Colombier 				aulBBD, tBBDLen, BIG_BLOCK_SIZE,
237*25b329d5SDavid du Colombier 				aucTmp, aulSectPage[tIndex], 2)) {
238f5736e95SDavid du Colombier 			continue;
239f5736e95SDavid du Colombier 		}
240f5736e95SDavid du Colombier 		tBytes = 2 + (size_t)usGetWord(0, aucTmp);
241f5736e95SDavid du Colombier 		NO_DBG_DEC(tBytes);
242f5736e95SDavid du Colombier 		/* Read the bytes */
243f5736e95SDavid du Colombier 		aucFpage = xmalloc(tBytes);
244f5736e95SDavid du Colombier 		if (!bReadBuffer(pFile, ulStartBlock,
245f5736e95SDavid du Colombier 				aulBBD, tBBDLen, BIG_BLOCK_SIZE,
246*25b329d5SDavid du Colombier 				aucFpage, aulSectPage[tIndex], tBytes)) {
247f5736e95SDavid du Colombier 			aucFpage = xfree(aucFpage);
248f5736e95SDavid du Colombier 			continue;
249f5736e95SDavid du Colombier 		}
250f5736e95SDavid du Colombier 		NO_DBG_PRINT_BLOCK(aucFpage, tBytes);
251f5736e95SDavid du Colombier 		/* Process the bytes */
252f5736e95SDavid du Colombier 		vGetDefaultSection(&tSection);
253f5736e95SDavid du Colombier 		vGet6SectionInfo(aucFpage + 2, tBytes - 2, &tSection);
254*25b329d5SDavid du Colombier 		vAdd2SectionInfoList(&tSection, aulCharPos[tIndex]);
255f5736e95SDavid du Colombier 		aucFpage = xfree(aucFpage);
256f5736e95SDavid du Colombier 	}
257*25b329d5SDavid du Colombier 	aulCharPos = xfree(aulCharPos);
258f5736e95SDavid du Colombier 	aulSectPage = xfree(aulSectPage);
259f5736e95SDavid du Colombier } /* end of vGet6SepInfo */
260f5736e95SDavid du Colombier 
261f5736e95SDavid du Colombier /*
262*25b329d5SDavid du Colombier  * Build the list with Header/Footer Information for Word 6/7 files
263*25b329d5SDavid du Colombier  */
264*25b329d5SDavid du Colombier void
vGet6HdrFtrInfo(FILE * pFile,ULONG ulStartBlock,const ULONG * aulBBD,size_t tBBDLen,const UCHAR * aucHeader)265*25b329d5SDavid du Colombier vGet6HdrFtrInfo(FILE *pFile, ULONG ulStartBlock,
266*25b329d5SDavid du Colombier 	const ULONG *aulBBD, size_t tBBDLen,
267*25b329d5SDavid du Colombier 	const UCHAR *aucHeader)
268*25b329d5SDavid du Colombier {
269*25b329d5SDavid du Colombier 	ULONG	*aulCharPos;
270*25b329d5SDavid du Colombier 	UCHAR	*aucBuffer;
271*25b329d5SDavid du Colombier 	ULONG	ulHdrFtrOffset, ulBeginHdrFtrInfo;
272*25b329d5SDavid du Colombier 	size_t	tHdrFtrInfoLen, tIndex, tOffset, tLen;
273*25b329d5SDavid du Colombier 
274*25b329d5SDavid du Colombier 	fail(pFile == NULL || aucHeader == NULL);
275*25b329d5SDavid du Colombier 	fail(ulStartBlock > MAX_BLOCKNUMBER && ulStartBlock != END_OF_CHAIN);
276*25b329d5SDavid du Colombier 	fail(aulBBD == NULL);
277*25b329d5SDavid du Colombier 
278*25b329d5SDavid du Colombier 	ulBeginHdrFtrInfo = ulGetLong(0xb0, aucHeader); /* fcPlcfhdd */
279*25b329d5SDavid du Colombier 	NO_DBG_HEX(ulBeginHdrFtrInfo);
280*25b329d5SDavid du Colombier 	tHdrFtrInfoLen = (size_t)ulGetLong(0xb4, aucHeader); /* lcbPlcfhdd */
281*25b329d5SDavid du Colombier 	NO_DBG_DEC(tHdrFtrInfoLen);
282*25b329d5SDavid du Colombier 	if (tHdrFtrInfoLen < 8) {
283*25b329d5SDavid du Colombier 		DBG_DEC_C(tHdrFtrInfoLen != 0, tHdrFtrInfoLen);
284*25b329d5SDavid du Colombier 		return;
285*25b329d5SDavid du Colombier 	}
286*25b329d5SDavid du Colombier 
287*25b329d5SDavid du Colombier 	aucBuffer = xmalloc(tHdrFtrInfoLen);
288*25b329d5SDavid du Colombier 	if (!bReadBuffer(pFile, ulStartBlock,
289*25b329d5SDavid du Colombier 			aulBBD, tBBDLen, BIG_BLOCK_SIZE,
290*25b329d5SDavid du Colombier 			aucBuffer, ulBeginHdrFtrInfo, tHdrFtrInfoLen)) {
291*25b329d5SDavid du Colombier 		aucBuffer = xfree(aucBuffer);
292*25b329d5SDavid du Colombier 		return;
293*25b329d5SDavid du Colombier 	}
294*25b329d5SDavid du Colombier 	NO_DBG_PRINT_BLOCK(aucBuffer, tHdrFtrInfoLen);
295*25b329d5SDavid du Colombier 
296*25b329d5SDavid du Colombier 	tLen = tHdrFtrInfoLen / 4 - 1;
297*25b329d5SDavid du Colombier 	/* Save the header/footer offsets */
298*25b329d5SDavid du Colombier 	aulCharPos = xcalloc(tLen, sizeof(ULONG));
299*25b329d5SDavid du Colombier 	for (tIndex = 0, tOffset = 0;
300*25b329d5SDavid du Colombier 	     tIndex < tLen;
301*25b329d5SDavid du Colombier 	     tIndex++, tOffset += 4) {
302*25b329d5SDavid du Colombier 		ulHdrFtrOffset = ulGetLong(tOffset, aucBuffer);
303*25b329d5SDavid du Colombier 		NO_DBG_HEX(ulHdrFtrOffset);
304*25b329d5SDavid du Colombier 		aulCharPos[tIndex] = ulHdrFtrOffset2CharPos(ulHdrFtrOffset);
305*25b329d5SDavid du Colombier 		NO_DBG_HEX(aulCharPos[tIndex]);
306*25b329d5SDavid du Colombier 	}
307*25b329d5SDavid du Colombier 	vCreat6HdrFtrInfoList(aulCharPos, tLen);
308*25b329d5SDavid du Colombier 	aulCharPos = xfree(aulCharPos);
309*25b329d5SDavid du Colombier 	aucBuffer = xfree(aucBuffer);
310*25b329d5SDavid du Colombier } /* end of vGet6HdrFtrInfo */
311*25b329d5SDavid du Colombier 
312*25b329d5SDavid du Colombier /*
313f5736e95SDavid du Colombier  * Translate the rowinfo to a member of the row_info enumeration
314f5736e95SDavid du Colombier  */
315f5736e95SDavid du Colombier row_info_enum
eGet6RowInfo(int iFodo,const UCHAR * aucGrpprl,int iBytes,row_block_type * pRow)316f5736e95SDavid du Colombier eGet6RowInfo(int iFodo,
317f5736e95SDavid du Colombier 	const UCHAR *aucGrpprl, int iBytes, row_block_type *pRow)
318f5736e95SDavid du Colombier {
319f5736e95SDavid du Colombier 	int	iFodoOff, iInfoLen;
320f5736e95SDavid du Colombier 	int	iIndex, iSize, iCol;
321f5736e95SDavid du Colombier 	int	iPosCurr, iPosPrev;
322f5736e95SDavid du Colombier 	USHORT	usTmp;
323f5736e95SDavid du Colombier 	BOOL	bFound24_0, bFound24_1, bFound25_0, bFound25_1, bFound190;
324f5736e95SDavid du Colombier 
325f5736e95SDavid du Colombier 	fail(iFodo < 0 || aucGrpprl == NULL || pRow == NULL);
326f5736e95SDavid du Colombier 
327f5736e95SDavid du Colombier 	iFodoOff = 0;
328f5736e95SDavid du Colombier 	bFound24_0 = FALSE;
329f5736e95SDavid du Colombier 	bFound24_1 = FALSE;
330f5736e95SDavid du Colombier 	bFound25_0 = FALSE;
331f5736e95SDavid du Colombier 	bFound25_1 = FALSE;
332f5736e95SDavid du Colombier 	bFound190 = FALSE;
333f5736e95SDavid du Colombier 	while (iBytes >= iFodoOff + 1) {
334f5736e95SDavid du Colombier 		iInfoLen = 0;
335f5736e95SDavid du Colombier 		switch (ucGetByte(iFodo + iFodoOff, aucGrpprl)) {
336*25b329d5SDavid du Colombier 		case  24:	/* fInTable */
337f5736e95SDavid du Colombier 			if (odd(ucGetByte(iFodo + iFodoOff + 1, aucGrpprl))) {
338f5736e95SDavid du Colombier 				bFound24_1 = TRUE;
339f5736e95SDavid du Colombier 			} else {
340f5736e95SDavid du Colombier 				bFound24_0 = TRUE;
341f5736e95SDavid du Colombier 			}
342f5736e95SDavid du Colombier 			break;
343f5736e95SDavid du Colombier 		case  25:	/* fTtp */
344f5736e95SDavid du Colombier 			if (odd(ucGetByte(iFodo + iFodoOff + 1, aucGrpprl))) {
345f5736e95SDavid du Colombier 				bFound25_1 = TRUE;
346f5736e95SDavid du Colombier 			} else {
347f5736e95SDavid du Colombier 				bFound25_0 = TRUE;
348f5736e95SDavid du Colombier 			}
349f5736e95SDavid du Colombier 			break;
350f5736e95SDavid du Colombier 		case 38:	/* brcTop */
351f5736e95SDavid du Colombier 			usTmp = usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
352f5736e95SDavid du Colombier 			usTmp &= 0x0018;
353f5736e95SDavid du Colombier 			NO_DBG_DEC(usTmp >> 3);
354f5736e95SDavid du Colombier 			if (usTmp == 0) {
355f5736e95SDavid du Colombier 				pRow->ucBorderInfo &= ~TABLE_BORDER_TOP;
356f5736e95SDavid du Colombier 			} else {
357f5736e95SDavid du Colombier 				pRow->ucBorderInfo |= TABLE_BORDER_TOP;
358f5736e95SDavid du Colombier 			}
359f5736e95SDavid du Colombier 			break;
360f5736e95SDavid du Colombier 		case 39:	/* brcLeft */
361f5736e95SDavid du Colombier 			usTmp = usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
362f5736e95SDavid du Colombier 			usTmp &= 0x0018;
363f5736e95SDavid du Colombier 			NO_DBG_DEC(usTmp >> 3);
364f5736e95SDavid du Colombier 			if (usTmp == 0) {
365f5736e95SDavid du Colombier 				pRow->ucBorderInfo &= ~TABLE_BORDER_LEFT;
366f5736e95SDavid du Colombier 			} else {
367f5736e95SDavid du Colombier 				pRow->ucBorderInfo |= TABLE_BORDER_LEFT;
368f5736e95SDavid du Colombier 			}
369f5736e95SDavid du Colombier 			break;
370f5736e95SDavid du Colombier 		case 40:	/* brcBottom */
371f5736e95SDavid du Colombier 			usTmp = usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
372f5736e95SDavid du Colombier 			usTmp &= 0x0018;
373f5736e95SDavid du Colombier 			NO_DBG_DEC(usTmp >> 3);
374f5736e95SDavid du Colombier 			if (usTmp == 0) {
375f5736e95SDavid du Colombier 				pRow->ucBorderInfo &= ~TABLE_BORDER_BOTTOM;
376f5736e95SDavid du Colombier 			} else {
377f5736e95SDavid du Colombier 				pRow->ucBorderInfo |= TABLE_BORDER_BOTTOM;
378f5736e95SDavid du Colombier 			}
379f5736e95SDavid du Colombier 			break;
380f5736e95SDavid du Colombier 		case 41:	/* brcRight */
381f5736e95SDavid du Colombier 			usTmp = usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
382f5736e95SDavid du Colombier 			usTmp &= 0x0018;
383f5736e95SDavid du Colombier 			NO_DBG_DEC(usTmp >> 3);
384f5736e95SDavid du Colombier 			if (usTmp == 0) {
385f5736e95SDavid du Colombier 				pRow->ucBorderInfo &= ~TABLE_BORDER_RIGHT;
386f5736e95SDavid du Colombier 			} else {
387f5736e95SDavid du Colombier 				pRow->ucBorderInfo |= TABLE_BORDER_RIGHT;
388f5736e95SDavid du Colombier 			}
389f5736e95SDavid du Colombier 			break;
390*25b329d5SDavid du Colombier 		case 188:	/* cDefTable10 */
391*25b329d5SDavid du Colombier 			DBG_MSG("188: sprmTDefTable10");
392*25b329d5SDavid du Colombier 			iSize = (int)usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
393*25b329d5SDavid du Colombier 			DBG_DEC(iSize);
394*25b329d5SDavid du Colombier 			break;
395f5736e95SDavid du Colombier 		case 190:	/* cDefTable */
396f5736e95SDavid du Colombier 			iSize = (int)usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
397f5736e95SDavid du Colombier 			if (iSize < 6 || iBytes < iFodoOff + 7) {
398f5736e95SDavid du Colombier 				DBG_DEC(iSize);
399f5736e95SDavid du Colombier 				DBG_DEC(iFodoOff);
400f5736e95SDavid du Colombier 				iInfoLen = 1;
401f5736e95SDavid du Colombier 				break;
402f5736e95SDavid du Colombier 			}
403f5736e95SDavid du Colombier 			iCol = (int)ucGetByte(iFodo + iFodoOff + 3, aucGrpprl);
404f5736e95SDavid du Colombier 			if (iCol < 1 ||
405f5736e95SDavid du Colombier 			    iBytes < iFodoOff + 3 + (iCol + 1) * 2) {
406f5736e95SDavid du Colombier 				DBG_DEC(iCol);
407f5736e95SDavid du Colombier 				DBG_DEC(iFodoOff);
408f5736e95SDavid du Colombier 				iInfoLen = 1;
409f5736e95SDavid du Colombier 				break;
410f5736e95SDavid du Colombier 			}
411f5736e95SDavid du Colombier 			if (iCol >= (int)elementsof(pRow->asColumnWidth)) {
412f5736e95SDavid du Colombier 				DBG_DEC(iCol);
413f5736e95SDavid du Colombier 				werr(1, "The number of columns is corrupt");
414f5736e95SDavid du Colombier 			}
415f5736e95SDavid du Colombier 			pRow->ucNumberOfColumns = (UCHAR)iCol;
416f5736e95SDavid du Colombier 			iPosPrev = (int)(short)usGetWord(
417f5736e95SDavid du Colombier 					iFodo + iFodoOff + 4,
418f5736e95SDavid du Colombier 					aucGrpprl);
419f5736e95SDavid du Colombier 			for (iIndex = 0; iIndex < iCol; iIndex++) {
420f5736e95SDavid du Colombier 				iPosCurr = (int)(short)usGetWord(
421f5736e95SDavid du Colombier 					iFodo + iFodoOff + 6 + iIndex * 2,
422f5736e95SDavid du Colombier 					aucGrpprl);
423f5736e95SDavid du Colombier 				pRow->asColumnWidth[iIndex] =
424f5736e95SDavid du Colombier 						(short)(iPosCurr - iPosPrev);
425f5736e95SDavid du Colombier 				iPosPrev = iPosCurr;
426f5736e95SDavid du Colombier 			}
427f5736e95SDavid du Colombier 			bFound190 = TRUE;
428f5736e95SDavid du Colombier 			break;
429f5736e95SDavid du Colombier 		default:
430f5736e95SDavid du Colombier 			break;
431f5736e95SDavid du Colombier 		}
432f5736e95SDavid du Colombier 		if (iInfoLen <= 0) {
433f5736e95SDavid du Colombier 			iInfoLen =
434f5736e95SDavid du Colombier 				iGet6InfoLength(iFodo + iFodoOff, aucGrpprl);
435f5736e95SDavid du Colombier 			fail(iInfoLen <= 0);
436f5736e95SDavid du Colombier 		}
437f5736e95SDavid du Colombier 		iFodoOff += iInfoLen;
438f5736e95SDavid du Colombier 	}
439*25b329d5SDavid du Colombier 
440*25b329d5SDavid du Colombier 	if (bFound25_1 && bFound190) {
441f5736e95SDavid du Colombier 		return found_end_of_row;
442f5736e95SDavid du Colombier 	}
443*25b329d5SDavid du Colombier 	if (bFound25_0 && !bFound190) {
444f5736e95SDavid du Colombier 		return found_not_end_of_row;
445f5736e95SDavid du Colombier 	}
446f5736e95SDavid du Colombier 	if (bFound24_1) {
447f5736e95SDavid du Colombier 		return found_a_cell;
448f5736e95SDavid du Colombier 	}
449f5736e95SDavid du Colombier 	if (bFound24_0) {
450f5736e95SDavid du Colombier 		return found_not_a_cell;
451f5736e95SDavid du Colombier 	}
452f5736e95SDavid du Colombier 	return found_nothing;
453f5736e95SDavid du Colombier } /* end of eGet6RowInfo */
454f5736e95SDavid du Colombier 
455f5736e95SDavid du Colombier /*
456f5736e95SDavid du Colombier  * Fill the style information block with information
457f5736e95SDavid du Colombier  * from a Word 6/7 file.
458f5736e95SDavid du Colombier  */
459f5736e95SDavid du Colombier void
vGet6StyleInfo(int iFodo,const UCHAR * aucGrpprl,int iBytes,style_block_type * pStyle)460f5736e95SDavid du Colombier vGet6StyleInfo(int iFodo,
461f5736e95SDavid du Colombier 	const UCHAR *aucGrpprl, int iBytes, style_block_type *pStyle)
462f5736e95SDavid du Colombier {
463f5736e95SDavid du Colombier 	int	iFodoOff, iInfoLen;
464f5736e95SDavid du Colombier 	int	iTmp, iDel, iAdd, iBefore;
465f5736e95SDavid du Colombier 	short	sTmp;
466f5736e95SDavid du Colombier 	UCHAR	ucTmp;
467f5736e95SDavid du Colombier 
468f5736e95SDavid du Colombier 	fail(iFodo < 0 || aucGrpprl == NULL || pStyle == NULL);
469f5736e95SDavid du Colombier 
470f5736e95SDavid du Colombier 	NO_DBG_DEC(pStyle->usIstd);
471f5736e95SDavid du Colombier 
472f5736e95SDavid du Colombier 	iFodoOff = 0;
473f5736e95SDavid du Colombier 	while (iBytes >= iFodoOff + 1) {
474f5736e95SDavid du Colombier 		iInfoLen = 0;
475f5736e95SDavid du Colombier 		switch (ucGetByte(iFodo + iFodoOff, aucGrpprl)) {
476f5736e95SDavid du Colombier 		case   2:	/* istd */
477f5736e95SDavid du Colombier 			sTmp = (short)ucGetByte(
478f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
479f5736e95SDavid du Colombier 			NO_DBG_DEC(sTmp);
480f5736e95SDavid du Colombier 			break;
481f5736e95SDavid du Colombier 		case   5:	/* jc */
482f5736e95SDavid du Colombier 			pStyle->ucAlignment = ucGetByte(
483f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
484f5736e95SDavid du Colombier 			break;
485f5736e95SDavid du Colombier 		case  12:	/* anld */
486f5736e95SDavid du Colombier 			iTmp = (int)ucGetByte(
487f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
488f5736e95SDavid du Colombier 			DBG_DEC_C(iTmp < 52, iTmp);
489f5736e95SDavid du Colombier 			if (iTmp >= 1) {
490f5736e95SDavid du Colombier 				pStyle->ucNFC = ucGetByte(
491f5736e95SDavid du Colombier 					iFodo + iFodoOff + 2, aucGrpprl);
492f5736e95SDavid du Colombier 			}
493f5736e95SDavid du Colombier 			if (pStyle->ucNFC != LIST_BULLETS && iTmp >= 2) {
494f5736e95SDavid du Colombier 				iBefore = (int)ucGetByte(
495f5736e95SDavid du Colombier 					iFodo + iFodoOff + 3, aucGrpprl);
496f5736e95SDavid du Colombier 			} else {
497f5736e95SDavid du Colombier 				iBefore = 0;
498f5736e95SDavid du Colombier 			}
499f5736e95SDavid du Colombier 			if (iTmp >= 12) {
500f5736e95SDavid du Colombier 				pStyle->usStartAt = usGetWord(
501f5736e95SDavid du Colombier 					iFodo + iFodoOff + 12, aucGrpprl);
502f5736e95SDavid du Colombier 			}
503f5736e95SDavid du Colombier 			if (iTmp >= iBefore + 21) {
504f5736e95SDavid du Colombier 				pStyle->usListChar = (USHORT)ucGetByte(
505f5736e95SDavid du Colombier 					iFodo + iFodoOff + iBefore + 22,
506f5736e95SDavid du Colombier 					aucGrpprl);
507f5736e95SDavid du Colombier 				NO_DBG_HEX(pStyle->usListChar);
508f5736e95SDavid du Colombier 			}
509f5736e95SDavid du Colombier 			break;
510f5736e95SDavid du Colombier 		case  13:	/* nLvlAnm */
511f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
512f5736e95SDavid du Colombier 			pStyle->ucNumLevel = ucTmp;
513f5736e95SDavid du Colombier 			pStyle->bNumPause =
514f5736e95SDavid du Colombier 				eGetNumType(ucTmp) == level_type_pause;
515f5736e95SDavid du Colombier 			break;
516f5736e95SDavid du Colombier 		case  15:	/* ChgTabsPapx */
517*25b329d5SDavid du Colombier 		case  23:	/* ChgTabs */
518f5736e95SDavid du Colombier 			iTmp = (int)ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
519f5736e95SDavid du Colombier 			if (iTmp < 2) {
520f5736e95SDavid du Colombier 				iInfoLen = 1;
521f5736e95SDavid du Colombier 				break;
522f5736e95SDavid du Colombier 			}
523f5736e95SDavid du Colombier 			NO_DBG_DEC(iTmp);
524f5736e95SDavid du Colombier 			iDel = (int)ucGetByte(iFodo + iFodoOff + 2, aucGrpprl);
525f5736e95SDavid du Colombier 			if (iTmp < 2 + 2 * iDel) {
526f5736e95SDavid du Colombier 				iInfoLen = 1;
527f5736e95SDavid du Colombier 				break;
528f5736e95SDavid du Colombier 			}
529f5736e95SDavid du Colombier 			NO_DBG_DEC(iDel);
530f5736e95SDavid du Colombier 			iAdd = (int)ucGetByte(
531f5736e95SDavid du Colombier 				iFodo + iFodoOff + 3 + 2 * iDel, aucGrpprl);
532f5736e95SDavid du Colombier 			if (iTmp < 2 + 2 * iDel + 2 * iAdd) {
533f5736e95SDavid du Colombier 				iInfoLen = 1;
534f5736e95SDavid du Colombier 				break;
535f5736e95SDavid du Colombier 			}
536f5736e95SDavid du Colombier 			NO_DBG_DEC(iAdd);
537f5736e95SDavid du Colombier 			break;
538f5736e95SDavid du Colombier 		case  16:	/* dxaRight */
539f5736e95SDavid du Colombier 			pStyle->sRightIndent = (short)usGetWord(
540f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
541f5736e95SDavid du Colombier 			NO_DBG_DEC(pStyle->sRightIndent);
542f5736e95SDavid du Colombier 			break;
543f5736e95SDavid du Colombier 		case  17:	/* dxaLeft */
544f5736e95SDavid du Colombier 			pStyle->sLeftIndent = (short)usGetWord(
545f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
546f5736e95SDavid du Colombier 			NO_DBG_DEC(pStyle->sLeftIndent);
547f5736e95SDavid du Colombier 			break;
548f5736e95SDavid du Colombier 		case  18:	/* Nest dxaLeft */
549f5736e95SDavid du Colombier 			sTmp = (short)usGetWord(
550f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
551f5736e95SDavid du Colombier 			pStyle->sLeftIndent += sTmp;
552f5736e95SDavid du Colombier 			if (pStyle->sLeftIndent < 0) {
553f5736e95SDavid du Colombier 				pStyle->sLeftIndent = 0;
554f5736e95SDavid du Colombier 			}
555f5736e95SDavid du Colombier 			NO_DBG_DEC(sTmp);
556f5736e95SDavid du Colombier 			NO_DBG_DEC(pStyle->sLeftIndent);
557f5736e95SDavid du Colombier 			break;
558f5736e95SDavid du Colombier 		case  19:	/* dxaLeft1 */
559f5736e95SDavid du Colombier 			pStyle->sLeftIndent1 = (short)usGetWord(
560f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
561f5736e95SDavid du Colombier 			NO_DBG_DEC(pStyle->sLeftIndent1);
562f5736e95SDavid du Colombier 			break;
563f5736e95SDavid du Colombier 		case  21:	/* dyaBefore */
564f5736e95SDavid du Colombier 			pStyle->usBeforeIndent = usGetWord(
565f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
566f5736e95SDavid du Colombier 			NO_DBG_DEC(pStyle->usBeforeIndent);
567f5736e95SDavid du Colombier 			break;
568f5736e95SDavid du Colombier 		case  22:	/* dyaAfter */
569f5736e95SDavid du Colombier 			pStyle->usAfterIndent = usGetWord(
570f5736e95SDavid du Colombier 					iFodo + iFodoOff + 1, aucGrpprl);
571f5736e95SDavid du Colombier 			NO_DBG_DEC(pStyle->usAfterIndent);
572f5736e95SDavid du Colombier 			break;
573f5736e95SDavid du Colombier 		default:
574f5736e95SDavid du Colombier 			break;
575f5736e95SDavid du Colombier 		}
576f5736e95SDavid du Colombier 		if (iInfoLen <= 0) {
577f5736e95SDavid du Colombier 			iInfoLen =
578f5736e95SDavid du Colombier 				iGet6InfoLength(iFodo + iFodoOff, aucGrpprl);
579f5736e95SDavid du Colombier 			fail(iInfoLen <= 0);
580f5736e95SDavid du Colombier 		}
581f5736e95SDavid du Colombier 		iFodoOff += iInfoLen;
582f5736e95SDavid du Colombier 	}
583f5736e95SDavid du Colombier } /* end of vGet6StyleInfo */
584f5736e95SDavid du Colombier 
585f5736e95SDavid du Colombier /*
586f5736e95SDavid du Colombier  * Build the lists with Paragraph Information for Word 6/7 files
587f5736e95SDavid du Colombier  */
588f5736e95SDavid du Colombier void
vGet6PapInfo(FILE * pFile,ULONG ulStartBlock,const ULONG * aulBBD,size_t tBBDLen,const UCHAR * aucHeader)589f5736e95SDavid du Colombier vGet6PapInfo(FILE *pFile, ULONG ulStartBlock,
590f5736e95SDavid du Colombier 	const ULONG *aulBBD, size_t tBBDLen,
591f5736e95SDavid du Colombier 	const UCHAR *aucHeader)
592f5736e95SDavid du Colombier {
593f5736e95SDavid du Colombier 	row_block_type		tRow;
594f5736e95SDavid du Colombier 	style_block_type	tStyle;
595f5736e95SDavid du Colombier 	USHORT	*ausParfPage;
596f5736e95SDavid du Colombier 	UCHAR	*aucBuffer;
597f5736e95SDavid du Colombier 	ULONG	ulCharPos, ulCharPosFirst, ulCharPosLast;
598f5736e95SDavid du Colombier 	ULONG	ulBeginParfInfo;
599f5736e95SDavid du Colombier 	size_t	tParfInfoLen, tParfPageNum, tOffset, tSize, tLenOld, tLen;
600*25b329d5SDavid du Colombier 	size_t	tIndex, tIndex2, tRun;
601*25b329d5SDavid du Colombier 	int	iFodo, iLen;
602f5736e95SDavid du Colombier 	row_info_enum	eRowInfo;
603f5736e95SDavid du Colombier 	USHORT	usParfFirstPage, usCount, usIstd;
604f5736e95SDavid du Colombier 	UCHAR	aucFpage[BIG_BLOCK_SIZE];
605f5736e95SDavid du Colombier 
606f5736e95SDavid du Colombier 	fail(pFile == NULL || aucHeader == NULL);
607f5736e95SDavid du Colombier 	fail(ulStartBlock > MAX_BLOCKNUMBER && ulStartBlock != END_OF_CHAIN);
608f5736e95SDavid du Colombier 	fail(aulBBD == NULL);
609f5736e95SDavid du Colombier 
610f5736e95SDavid du Colombier 	ulBeginParfInfo = ulGetLong(0xc0, aucHeader); /* fcPlcfbtePapx */
611f5736e95SDavid du Colombier 	NO_DBG_HEX(ulBeginParfInfo);
612f5736e95SDavid du Colombier 	tParfInfoLen = (size_t)ulGetLong(0xc4, aucHeader); /* lcbPlcfbtePapx */
613f5736e95SDavid du Colombier 	NO_DBG_DEC(tParfInfoLen);
614f5736e95SDavid du Colombier 	if (tParfInfoLen < 4) {
615f5736e95SDavid du Colombier 		DBG_DEC(tParfInfoLen);
616f5736e95SDavid du Colombier 		return;
617f5736e95SDavid du Colombier 	}
618f5736e95SDavid du Colombier 
619f5736e95SDavid du Colombier 	aucBuffer = xmalloc(tParfInfoLen);
620f5736e95SDavid du Colombier 	if (!bReadBuffer(pFile, ulStartBlock,
621f5736e95SDavid du Colombier 			aulBBD, tBBDLen, BIG_BLOCK_SIZE,
622f5736e95SDavid du Colombier 			aucBuffer, ulBeginParfInfo, tParfInfoLen)) {
623f5736e95SDavid du Colombier 		aucBuffer = xfree(aucBuffer);
624f5736e95SDavid du Colombier 		return;
625f5736e95SDavid du Colombier 	}
626f5736e95SDavid du Colombier 	NO_DBG_PRINT_BLOCK(aucBuffer, tParfInfoLen);
627f5736e95SDavid du Colombier 
628f5736e95SDavid du Colombier 	tLen = (tParfInfoLen - 4) / 6;
629f5736e95SDavid du Colombier 	ausParfPage = xcalloc(tLen, sizeof(USHORT));
630*25b329d5SDavid du Colombier 	for (tIndex = 0, tOffset = (tLen + 1) * 4;
631*25b329d5SDavid du Colombier 	     tIndex < tLen;
632*25b329d5SDavid du Colombier 	     tIndex++, tOffset += 2) {
633*25b329d5SDavid du Colombier 		 ausParfPage[tIndex] = usGetWord(tOffset, aucBuffer);
634*25b329d5SDavid du Colombier 		 NO_DBG_DEC(ausParfPage[tIndex]);
635f5736e95SDavid du Colombier 	}
636f5736e95SDavid du Colombier 	DBG_HEX(ulGetLong(0, aucBuffer));
637f5736e95SDavid du Colombier 	aucBuffer = xfree(aucBuffer);
638f5736e95SDavid du Colombier 	tParfPageNum = (size_t)usGetWord(0x190, aucHeader); /* cpnBtePap */
639f5736e95SDavid du Colombier 	DBG_DEC(tParfPageNum);
640f5736e95SDavid du Colombier 	if (tLen < tParfPageNum) {
641f5736e95SDavid du Colombier 		/* Replace ParfPage by a longer version */
642f5736e95SDavid du Colombier 		tLenOld = tLen;
643f5736e95SDavid du Colombier 		usParfFirstPage = usGetWord(0x18c, aucHeader); /* pnPapFirst */
644f5736e95SDavid du Colombier 		DBG_DEC(usParfFirstPage);
645f5736e95SDavid du Colombier 		tLen += tParfPageNum - 1;
646f5736e95SDavid du Colombier 		tSize = tLen * sizeof(USHORT);
647f5736e95SDavid du Colombier 		ausParfPage = xrealloc(ausParfPage, tSize);
648f5736e95SDavid du Colombier 		/* Add new values */
649f5736e95SDavid du Colombier 		usCount = usParfFirstPage + 1;
650*25b329d5SDavid du Colombier 		for (tIndex = tLenOld; tIndex < tLen; tIndex++) {
651*25b329d5SDavid du Colombier 			ausParfPage[tIndex] = usCount;
652*25b329d5SDavid du Colombier 			NO_DBG_DEC(ausParfPage[tIndex]);
653f5736e95SDavid du Colombier 			usCount++;
654f5736e95SDavid du Colombier 		}
655f5736e95SDavid du Colombier 	}
656f5736e95SDavid du Colombier 
657f5736e95SDavid du Colombier 	(void)memset(&tRow, 0, sizeof(tRow));
658f5736e95SDavid du Colombier 	ulCharPosFirst = CP_INVALID;
659*25b329d5SDavid du Colombier 	for (tIndex = 0; tIndex < tLen; tIndex++) {
660f5736e95SDavid du Colombier 		if (!bReadBuffer(pFile, ulStartBlock,
661f5736e95SDavid du Colombier 				aulBBD, tBBDLen, BIG_BLOCK_SIZE,
662f5736e95SDavid du Colombier 				aucFpage,
663*25b329d5SDavid du Colombier 				(ULONG)ausParfPage[tIndex] * BIG_BLOCK_SIZE,
664f5736e95SDavid du Colombier 				BIG_BLOCK_SIZE)) {
665f5736e95SDavid du Colombier 			break;
666f5736e95SDavid du Colombier 		}
667*25b329d5SDavid du Colombier 		tRun = (size_t)ucGetByte(0x1ff, aucFpage);
668*25b329d5SDavid du Colombier 		NO_DBG_DEC(tRun);
669*25b329d5SDavid du Colombier 		for (tIndex2 = 0; tIndex2 < tRun; tIndex2++) {
670*25b329d5SDavid du Colombier 			NO_DBG_HEX(ulGetLong(tIndex2 * 4, aucFpage));
671f5736e95SDavid du Colombier 			iFodo = 2 * (int)ucGetByte(
672*25b329d5SDavid du Colombier 				(tRun + 1) * 4 + tIndex2 * 7, aucFpage);
673f5736e95SDavid du Colombier 			if (iFodo <= 0) {
674f5736e95SDavid du Colombier 				continue;
675f5736e95SDavid du Colombier 			}
676f5736e95SDavid du Colombier 
677f5736e95SDavid du Colombier 			iLen = 2 * (int)ucGetByte(iFodo, aucFpage);
678f5736e95SDavid du Colombier 
679f5736e95SDavid du Colombier 			usIstd = (USHORT)ucGetByte(iFodo + 1, aucFpage);
680f5736e95SDavid du Colombier 			vFillStyleFromStylesheet(usIstd, &tStyle);
681f5736e95SDavid du Colombier 			vGet6StyleInfo(iFodo, aucFpage + 3, iLen - 3, &tStyle);
682*25b329d5SDavid du Colombier 			ulCharPos = ulGetLong(tIndex2 * 4, aucFpage);
683f5736e95SDavid du Colombier 			NO_DBG_HEX(ulCharPos);
684*25b329d5SDavid du Colombier 			tStyle.ulFileOffset = ulCharPos2FileOffsetX(
685*25b329d5SDavid du Colombier 				ulCharPos, &tStyle.eListID);
686f5736e95SDavid du Colombier 			vAdd2StyleInfoList(&tStyle);
687f5736e95SDavid du Colombier 
688f5736e95SDavid du Colombier 			eRowInfo = eGet6RowInfo(iFodo,
689f5736e95SDavid du Colombier 					aucFpage + 3, iLen - 3, &tRow);
690f5736e95SDavid du Colombier 			switch(eRowInfo) {
691f5736e95SDavid du Colombier 			case found_a_cell:
692f5736e95SDavid du Colombier 				if (ulCharPosFirst != CP_INVALID) {
693f5736e95SDavid du Colombier 					break;
694f5736e95SDavid du Colombier 				}
695f5736e95SDavid du Colombier 				ulCharPosFirst = ulGetLong(
696*25b329d5SDavid du Colombier 						tIndex2 * 4, aucFpage);
697f5736e95SDavid du Colombier 				NO_DBG_HEX(ulCharPosFirst);
698f5736e95SDavid du Colombier 				tRow.ulCharPosStart = ulCharPosFirst;
699f5736e95SDavid du Colombier 				tRow.ulFileOffsetStart =
700f5736e95SDavid du Colombier 					ulCharPos2FileOffset(ulCharPosFirst);
701f5736e95SDavid du Colombier 				DBG_HEX_C(tRow.ulFileOffsetStart == FC_INVALID,
702f5736e95SDavid du Colombier 							ulCharPosFirst);
703f5736e95SDavid du Colombier 				break;
704f5736e95SDavid du Colombier 			case found_end_of_row:
705f5736e95SDavid du Colombier 				ulCharPosLast = ulGetLong(
706*25b329d5SDavid du Colombier 						tIndex2 * 4, aucFpage);
707f5736e95SDavid du Colombier 				NO_DBG_HEX(ulCharPosLast);
708f5736e95SDavid du Colombier 				tRow.ulCharPosEnd = ulCharPosLast;
709*25b329d5SDavid du Colombier 				tRow.ulFileOffsetEnd =
710*25b329d5SDavid du Colombier 					ulCharPos2FileOffset(ulCharPosLast);
711f5736e95SDavid du Colombier 				DBG_HEX_C(tRow.ulFileOffsetEnd == FC_INVALID,
712f5736e95SDavid du Colombier 							ulCharPosLast);
713f5736e95SDavid du Colombier 				vAdd2RowInfoList(&tRow);
714f5736e95SDavid du Colombier 				(void)memset(&tRow, 0, sizeof(tRow));
715f5736e95SDavid du Colombier 				ulCharPosFirst = CP_INVALID;
716f5736e95SDavid du Colombier 				break;
717f5736e95SDavid du Colombier 			case found_nothing:
718f5736e95SDavid du Colombier 				break;
719f5736e95SDavid du Colombier 			default:
720f5736e95SDavid du Colombier 				DBG_DEC(eRowInfo);
721f5736e95SDavid du Colombier 				break;
722f5736e95SDavid du Colombier 			}
723f5736e95SDavid du Colombier 		}
724f5736e95SDavid du Colombier 	}
725f5736e95SDavid du Colombier 	ausParfPage = xfree(ausParfPage);
726f5736e95SDavid du Colombier } /* end of vGet6PapInfo */
727f5736e95SDavid du Colombier 
728f5736e95SDavid du Colombier /*
729f5736e95SDavid du Colombier  * Fill the font information block with information
730f5736e95SDavid du Colombier  * from a Word 6/7 file.
731f5736e95SDavid du Colombier  * Returns TRUE when successful, otherwise FALSE
732f5736e95SDavid du Colombier  */
733f5736e95SDavid du Colombier void
vGet6FontInfo(int iFodo,USHORT usIstd,const UCHAR * aucGrpprl,int iBytes,font_block_type * pFont)734f5736e95SDavid du Colombier vGet6FontInfo(int iFodo, USHORT usIstd,
735f5736e95SDavid du Colombier 	const UCHAR *aucGrpprl, int iBytes, font_block_type *pFont)
736f5736e95SDavid du Colombier {
737f5736e95SDavid du Colombier 	long	lTmp;
738f5736e95SDavid du Colombier 	int	iFodoOff, iInfoLen;
739f5736e95SDavid du Colombier 	USHORT	usTmp;
740f5736e95SDavid du Colombier 	UCHAR	ucTmp;
741f5736e95SDavid du Colombier 
742*25b329d5SDavid du Colombier 	TRACE_MSG("vGet6FontInfo");
743*25b329d5SDavid du Colombier 
744f5736e95SDavid du Colombier 	fail(iFodo < 0 || aucGrpprl == NULL || pFont == NULL);
745f5736e95SDavid du Colombier 
746f5736e95SDavid du Colombier 	iFodoOff = 0;
747f5736e95SDavid du Colombier 	while (iBytes >= iFodoOff + 1) {
748f5736e95SDavid du Colombier 		switch (ucGetByte(iFodo + iFodoOff, aucGrpprl)) {
749f5736e95SDavid du Colombier 		case  65:	/* fRMarkDel */
750f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
751f5736e95SDavid du Colombier 			if (ucTmp == 0) {
752f5736e95SDavid du Colombier 				pFont->usFontStyle &= ~FONT_MARKDEL;
753f5736e95SDavid du Colombier 			} else {
754f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_MARKDEL;
755f5736e95SDavid du Colombier 			}
756f5736e95SDavid du Colombier 			break;
757f5736e95SDavid du Colombier 		case  80:	/* cIstd */
758f5736e95SDavid du Colombier 			usTmp = usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
759f5736e95SDavid du Colombier 			NO_DBG_DEC(usTmp);
760f5736e95SDavid du Colombier 			break;
761f5736e95SDavid du Colombier 		case  82:	/* cDefault */
762f5736e95SDavid du Colombier 			pFont->usFontStyle &= FONT_HIDDEN;
763f5736e95SDavid du Colombier 			pFont->ucFontColor = FONT_COLOR_DEFAULT;
764f5736e95SDavid du Colombier 			break;
765f5736e95SDavid du Colombier 		case  83:	/* cPlain */
766f5736e95SDavid du Colombier 			DBG_MSG("83: cPlain");
767f5736e95SDavid du Colombier 			vFillFontFromStylesheet(usIstd, pFont);
768f5736e95SDavid du Colombier 			break;
769f5736e95SDavid du Colombier 		case  85:	/* fBold */
770f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
771f5736e95SDavid du Colombier 			switch (ucTmp) {
772f5736e95SDavid du Colombier 			case   0:	/* Unset */
773f5736e95SDavid du Colombier 				pFont->usFontStyle &= ~FONT_BOLD;
774f5736e95SDavid du Colombier 				break;
775f5736e95SDavid du Colombier 			case   1:	/* Set */
776f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_BOLD;
777f5736e95SDavid du Colombier 				break;
778f5736e95SDavid du Colombier 			case 128:	/* Unchanged */
779f5736e95SDavid du Colombier 				break;
780f5736e95SDavid du Colombier 			case 129:	/* Negation */
781f5736e95SDavid du Colombier 				pFont->usFontStyle ^= FONT_BOLD;
782f5736e95SDavid du Colombier 				break;
783f5736e95SDavid du Colombier 			default:
784f5736e95SDavid du Colombier 				DBG_DEC(ucTmp);
785f5736e95SDavid du Colombier 				DBG_FIXME();
786f5736e95SDavid du Colombier 				break;
787f5736e95SDavid du Colombier 			}
788f5736e95SDavid du Colombier 			break;
789f5736e95SDavid du Colombier 		case  86:	/* fItalic */
790f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
791f5736e95SDavid du Colombier 			switch (ucTmp) {
792f5736e95SDavid du Colombier 			case   0:	/* Unset */
793f5736e95SDavid du Colombier 				pFont->usFontStyle &= ~FONT_ITALIC;
794f5736e95SDavid du Colombier 				break;
795f5736e95SDavid du Colombier 			case   1:	/* Set */
796f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_ITALIC;
797f5736e95SDavid du Colombier 				break;
798f5736e95SDavid du Colombier 			case 128:	/* Unchanged */
799f5736e95SDavid du Colombier 				break;
800f5736e95SDavid du Colombier 			case 129:	/* Negation */
801f5736e95SDavid du Colombier 				pFont->usFontStyle ^= FONT_ITALIC;
802f5736e95SDavid du Colombier 				break;
803f5736e95SDavid du Colombier 			default:
804f5736e95SDavid du Colombier 				DBG_DEC(ucTmp);
805f5736e95SDavid du Colombier 				DBG_FIXME();
806f5736e95SDavid du Colombier 				break;
807f5736e95SDavid du Colombier 			}
808f5736e95SDavid du Colombier 			break;
809f5736e95SDavid du Colombier 		case  87:	/* fStrike */
810f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
811f5736e95SDavid du Colombier 			switch (ucTmp) {
812f5736e95SDavid du Colombier 			case   0:	/* Unset */
813f5736e95SDavid du Colombier 				pFont->usFontStyle &= ~FONT_STRIKE;
814f5736e95SDavid du Colombier 				break;
815f5736e95SDavid du Colombier 			case   1:	/* Set */
816f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_STRIKE;
817f5736e95SDavid du Colombier 				break;
818f5736e95SDavid du Colombier 			case 128:	/* Unchanged */
819f5736e95SDavid du Colombier 				break;
820f5736e95SDavid du Colombier 			case 129:	/* Negation */
821f5736e95SDavid du Colombier 				pFont->usFontStyle ^= FONT_STRIKE;
822f5736e95SDavid du Colombier 				break;
823f5736e95SDavid du Colombier 			default:
824f5736e95SDavid du Colombier 				DBG_DEC(ucTmp);
825f5736e95SDavid du Colombier 				DBG_FIXME();
826f5736e95SDavid du Colombier 				break;
827f5736e95SDavid du Colombier 			}
828f5736e95SDavid du Colombier 			break;
829f5736e95SDavid du Colombier 		case  90:	/* fSmallCaps */
830f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
831f5736e95SDavid du Colombier 			switch (ucTmp) {
832f5736e95SDavid du Colombier 			case   0:	/* Unset */
833f5736e95SDavid du Colombier 				pFont->usFontStyle &= ~FONT_SMALL_CAPITALS;
834f5736e95SDavid du Colombier 				break;
835f5736e95SDavid du Colombier 			case   1:	/* Set */
836f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_SMALL_CAPITALS;
837f5736e95SDavid du Colombier 				break;
838f5736e95SDavid du Colombier 			case 128:	/* Unchanged */
839f5736e95SDavid du Colombier 				break;
840f5736e95SDavid du Colombier 			case 129:	/* Negation */
841f5736e95SDavid du Colombier 				pFont->usFontStyle ^= FONT_SMALL_CAPITALS;
842f5736e95SDavid du Colombier 				break;
843f5736e95SDavid du Colombier 			default:
844f5736e95SDavid du Colombier 				DBG_DEC(ucTmp);
845f5736e95SDavid du Colombier 				DBG_FIXME();
846f5736e95SDavid du Colombier 				break;
847f5736e95SDavid du Colombier 			}
848f5736e95SDavid du Colombier 			break;
849f5736e95SDavid du Colombier 		case  91:	/* fCaps */
850f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
851f5736e95SDavid du Colombier 			switch (ucTmp) {
852f5736e95SDavid du Colombier 			case   0:	/* Unset */
853f5736e95SDavid du Colombier 				pFont->usFontStyle &= ~FONT_CAPITALS;
854f5736e95SDavid du Colombier 				break;
855f5736e95SDavid du Colombier 			case   1:	/* Set */
856f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_CAPITALS;
857f5736e95SDavid du Colombier 				break;
858f5736e95SDavid du Colombier 			case 128:	/* Unchanged */
859f5736e95SDavid du Colombier 				break;
860f5736e95SDavid du Colombier 			case 129:	/* Negation */
861f5736e95SDavid du Colombier 				pFont->usFontStyle ^= FONT_CAPITALS;
862f5736e95SDavid du Colombier 				break;
863f5736e95SDavid du Colombier 			default:
864f5736e95SDavid du Colombier 				DBG_DEC(ucTmp);
865f5736e95SDavid du Colombier 				DBG_FIXME();
866f5736e95SDavid du Colombier 				break;
867f5736e95SDavid du Colombier 			}
868f5736e95SDavid du Colombier 			break;
869f5736e95SDavid du Colombier 		case  92:	/* fVanish */
870f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
871f5736e95SDavid du Colombier 			switch (ucTmp) {
872f5736e95SDavid du Colombier 			case   0:	/* Unset */
873f5736e95SDavid du Colombier 				pFont->usFontStyle &= ~FONT_HIDDEN;
874f5736e95SDavid du Colombier 				break;
875f5736e95SDavid du Colombier 			case   1:	/* Set */
876f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_HIDDEN;
877f5736e95SDavid du Colombier 				break;
878f5736e95SDavid du Colombier 			case 128:	/* Unchanged */
879f5736e95SDavid du Colombier 				break;
880f5736e95SDavid du Colombier 			case 129:	/* Negation */
881f5736e95SDavid du Colombier 				pFont->usFontStyle ^= FONT_HIDDEN;
882f5736e95SDavid du Colombier 				break;
883f5736e95SDavid du Colombier 			default:
884f5736e95SDavid du Colombier 				DBG_DEC(ucTmp);
885f5736e95SDavid du Colombier 				DBG_FIXME();
886f5736e95SDavid du Colombier 				break;
887f5736e95SDavid du Colombier 			}
888f5736e95SDavid du Colombier 			break;
889f5736e95SDavid du Colombier 		case  93:	/* cFtc */
890f5736e95SDavid du Colombier 			usTmp = usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
891f5736e95SDavid du Colombier 			if (usTmp <= (USHORT)UCHAR_MAX) {
892f5736e95SDavid du Colombier 				pFont->ucFontNumber = (UCHAR)usTmp;
893f5736e95SDavid du Colombier 			} else {
894*25b329d5SDavid du Colombier 				DBG_DEC(usTmp);
895*25b329d5SDavid du Colombier 				DBG_FIXME();
896f5736e95SDavid du Colombier 				pFont->ucFontNumber = 0;
897f5736e95SDavid du Colombier 			}
898f5736e95SDavid du Colombier 			break;
899f5736e95SDavid du Colombier 		case  94:	/* cKul */
900f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
901f5736e95SDavid du Colombier 			if (ucTmp == 0 || ucTmp == 5) {
902f5736e95SDavid du Colombier 				pFont->usFontStyle &= ~FONT_UNDERLINE;
903f5736e95SDavid du Colombier 			} else {
904f5736e95SDavid du Colombier 				NO_DBG_MSG("Underline text");
905f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_UNDERLINE;
906f5736e95SDavid du Colombier 				if (ucTmp == 6) {
907f5736e95SDavid du Colombier 					DBG_MSG("Bold text");
908f5736e95SDavid du Colombier 					pFont->usFontStyle |= FONT_BOLD;
909f5736e95SDavid du Colombier 				}
910f5736e95SDavid du Colombier 			}
911f5736e95SDavid du Colombier 			break;
912f5736e95SDavid du Colombier 		case  95:	/* cHps, cHpsPos */
913f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
914*25b329d5SDavid du Colombier 			DBG_DEC(ucTmp);
915f5736e95SDavid du Colombier 			if (ucTmp != 0) {
916f5736e95SDavid du Colombier 				pFont->usFontSize = (USHORT)ucTmp;
917f5736e95SDavid du Colombier 			}
918*25b329d5SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 2, aucGrpprl);
919f5736e95SDavid du Colombier 			DBG_DEC(ucTmp);
920f5736e95SDavid du Colombier 			break;
921f5736e95SDavid du Colombier 		case  98:	/* cIco */
922f5736e95SDavid du Colombier 			pFont->ucFontColor =
923f5736e95SDavid du Colombier 				ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
924f5736e95SDavid du Colombier 			break;
925f5736e95SDavid du Colombier 		case  99:	/* cHps */
926f5736e95SDavid du Colombier 			pFont->usFontSize =
927f5736e95SDavid du Colombier 				usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
928f5736e95SDavid du Colombier 			break;
929*25b329d5SDavid du Colombier 		case 100:	/* cHpsInc */
930*25b329d5SDavid du Colombier 			DBG_MSG("100: sprmCHpsInc");
931*25b329d5SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
932*25b329d5SDavid du Colombier 			DBG_DEC(ucTmp);
933*25b329d5SDavid du Colombier 			break;
934*25b329d5SDavid du Colombier 		case 103:	/* cMajority */
935*25b329d5SDavid du Colombier 			DBG_MSG("103: sprmCMajority");
936*25b329d5SDavid du Colombier 			break;
937f5736e95SDavid du Colombier 		case 104:	/* cIss */
938f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
939f5736e95SDavid du Colombier 			ucTmp &= 0x07;
940f5736e95SDavid du Colombier 			if (ucTmp == 1) {
941f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_SUPERSCRIPT;
942f5736e95SDavid du Colombier 				NO_DBG_MSG("Superscript");
943f5736e95SDavid du Colombier 			} else if (ucTmp == 2) {
944f5736e95SDavid du Colombier 				pFont->usFontStyle |= FONT_SUBSCRIPT;
945f5736e95SDavid du Colombier 				NO_DBG_MSG("Subscript");
946f5736e95SDavid du Colombier 			}
947f5736e95SDavid du Colombier 			break;
948*25b329d5SDavid du Colombier 		case 106:	/* cHpsInc1 */
949f5736e95SDavid du Colombier 			usTmp = usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
950f5736e95SDavid du Colombier 			lTmp = (long)pFont->usFontSize + (long)usTmp;
951f5736e95SDavid du Colombier 			if (lTmp < 8) {
952f5736e95SDavid du Colombier 				pFont->usFontSize = 8;
953f5736e95SDavid du Colombier 			} else if (lTmp > 32766) {
954f5736e95SDavid du Colombier 				pFont->usFontSize = 32766;
955f5736e95SDavid du Colombier 			} else {
956f5736e95SDavid du Colombier 				pFont->usFontSize = (USHORT)lTmp;
957f5736e95SDavid du Colombier 			}
958f5736e95SDavid du Colombier 			break;
959*25b329d5SDavid du Colombier 		case 108:	/* cMajority50 */
960*25b329d5SDavid du Colombier 			DBG_MSG("108: sprmCMajority50");
961*25b329d5SDavid du Colombier 			break;
962*25b329d5SDavid du Colombier 		case 109:	/* cHpsMul */
963*25b329d5SDavid du Colombier 			DBG_MSG("109: sprmCHpsMul");
964*25b329d5SDavid du Colombier 			usTmp = usGetWord(iFodo + iFodoOff + 1, aucGrpprl);
965*25b329d5SDavid du Colombier 			DBG_DEC(usTmp);
966*25b329d5SDavid du Colombier 			break;
967f5736e95SDavid du Colombier 		default:
968f5736e95SDavid du Colombier 			break;
969f5736e95SDavid du Colombier 		}
970f5736e95SDavid du Colombier 		iInfoLen = iGet6InfoLength(iFodo + iFodoOff, aucGrpprl);
971f5736e95SDavid du Colombier 		fail(iInfoLen <= 0);
972f5736e95SDavid du Colombier 		iFodoOff += iInfoLen;
973f5736e95SDavid du Colombier 	}
974f5736e95SDavid du Colombier } /* end of vGet6FontInfo */
975f5736e95SDavid du Colombier 
976f5736e95SDavid du Colombier /*
977f5736e95SDavid du Colombier  * Fill the picture information block with information
978f5736e95SDavid du Colombier  * from a Word 6/7 file.
979f5736e95SDavid du Colombier  * Returns TRUE when successful, otherwise FALSE
980f5736e95SDavid du Colombier  */
981f5736e95SDavid du Colombier static BOOL
bGet6PicInfo(int iFodo,const UCHAR * aucGrpprl,int iBytes,picture_block_type * pPicture)982f5736e95SDavid du Colombier bGet6PicInfo(int iFodo,
983f5736e95SDavid du Colombier 	const UCHAR *aucGrpprl, int iBytes, picture_block_type *pPicture)
984f5736e95SDavid du Colombier {
985f5736e95SDavid du Colombier 	int	iFodoOff, iInfoLen;
986f5736e95SDavid du Colombier 	BOOL	bFound;
987f5736e95SDavid du Colombier 	UCHAR	ucTmp;
988f5736e95SDavid du Colombier 
989*25b329d5SDavid du Colombier 	TRACE_MSG("vGet6PicInfo");
990*25b329d5SDavid du Colombier 
991f5736e95SDavid du Colombier 	fail(iFodo < 0 || aucGrpprl == NULL || pPicture == NULL);
992f5736e95SDavid du Colombier 
993f5736e95SDavid du Colombier 	iFodoOff = 0;
994f5736e95SDavid du Colombier 	bFound = FALSE;
995f5736e95SDavid du Colombier 	while (iBytes >= iFodoOff + 1) {
996f5736e95SDavid du Colombier 		switch (ucGetByte(iFodo + iFodoOff, aucGrpprl)) {
997f5736e95SDavid du Colombier 		case  68:	/* fcPic */
998f5736e95SDavid du Colombier 			pPicture->ulPictureOffset = ulGetLong(
999f5736e95SDavid du Colombier 					iFodo + iFodoOff + 2, aucGrpprl);
1000f5736e95SDavid du Colombier 			bFound = TRUE;
1001f5736e95SDavid du Colombier 			break;
1002*25b329d5SDavid du Colombier #if 0
1003*25b329d5SDavid du Colombier 		case  71:	/* fData */
1004f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
1005f5736e95SDavid du Colombier 			if (ucTmp == 0x01) {
1006f5736e95SDavid du Colombier 				/* Not a picture, but a form field */
1007f5736e95SDavid du Colombier 				return FALSE;
1008f5736e95SDavid du Colombier 			}
1009f5736e95SDavid du Colombier 			DBG_DEC_C(ucTmp != 0, ucTmp);
1010f5736e95SDavid du Colombier 			break;
1011*25b329d5SDavid du Colombier #endif
1012f5736e95SDavid du Colombier 		case  75:	/* fOle2 */
1013f5736e95SDavid du Colombier 			ucTmp = ucGetByte(iFodo + iFodoOff + 1, aucGrpprl);
1014f5736e95SDavid du Colombier 			if (ucTmp == 0x01) {
1015f5736e95SDavid du Colombier 				/* Not a picture, but an OLE object */
1016f5736e95SDavid du Colombier 				return FALSE;
1017f5736e95SDavid du Colombier 			}
1018f5736e95SDavid du Colombier 			DBG_DEC_C(ucTmp != 0, ucTmp);
1019f5736e95SDavid du Colombier 			break;
1020f5736e95SDavid du Colombier 		default:
1021f5736e95SDavid du Colombier 			break;
1022f5736e95SDavid du Colombier 		}
1023f5736e95SDavid du Colombier 		iInfoLen = iGet6InfoLength(iFodo + iFodoOff, aucGrpprl);
1024f5736e95SDavid du Colombier 		fail(iInfoLen <= 0);
1025f5736e95SDavid du Colombier 		iFodoOff += iInfoLen;
1026f5736e95SDavid du Colombier 	}
1027f5736e95SDavid du Colombier 	return bFound;
1028f5736e95SDavid du Colombier } /* end of bGet6PicInfo */
1029f5736e95SDavid du Colombier 
1030f5736e95SDavid du Colombier /*
1031f5736e95SDavid du Colombier  * Build the lists with Character Information for Word 6/7 files
1032f5736e95SDavid du Colombier  */
1033f5736e95SDavid du Colombier void
vGet6ChrInfo(FILE * pFile,ULONG ulStartBlock,const ULONG * aulBBD,size_t tBBDLen,const UCHAR * aucHeader)1034f5736e95SDavid du Colombier vGet6ChrInfo(FILE *pFile, ULONG ulStartBlock,
1035f5736e95SDavid du Colombier 	const ULONG *aulBBD, size_t tBBDLen, const UCHAR *aucHeader)
1036f5736e95SDavid du Colombier {
1037f5736e95SDavid du Colombier 	font_block_type		tFont;
1038f5736e95SDavid du Colombier 	picture_block_type	tPicture;
1039f5736e95SDavid du Colombier 	USHORT	*ausCharPage;
1040f5736e95SDavid du Colombier 	UCHAR	*aucBuffer;
1041f5736e95SDavid du Colombier 	ULONG	ulFileOffset, ulCharPos, ulBeginCharInfo;
1042f5736e95SDavid du Colombier 	size_t	tCharInfoLen, tOffset, tSize, tLenOld, tLen, tCharPageNum;
1043*25b329d5SDavid du Colombier 	size_t	tIndex, tIndex2, tRun;
1044*25b329d5SDavid du Colombier 	int	iFodo, iLen;
1045f5736e95SDavid du Colombier 	USHORT	usCharFirstPage, usCount, usIstd;
1046f5736e95SDavid du Colombier 	UCHAR	aucFpage[BIG_BLOCK_SIZE];
1047f5736e95SDavid du Colombier 
1048f5736e95SDavid du Colombier 	fail(pFile == NULL || aucHeader == NULL);
1049f5736e95SDavid du Colombier 	fail(ulStartBlock > MAX_BLOCKNUMBER && ulStartBlock != END_OF_CHAIN);
1050f5736e95SDavid du Colombier 	fail(aulBBD == NULL);
1051f5736e95SDavid du Colombier 
1052f5736e95SDavid du Colombier 	ulBeginCharInfo = ulGetLong(0xb8, aucHeader); /* fcPlcfbteChpx */
1053f5736e95SDavid du Colombier 	NO_DBG_HEX(lBeginCharInfo);
1054f5736e95SDavid du Colombier 	tCharInfoLen = (size_t)ulGetLong(0xbc, aucHeader); /* lcbPlcfbteChpx */
1055f5736e95SDavid du Colombier 	NO_DBG_DEC(tCharInfoLen);
1056f5736e95SDavid du Colombier 	if (tCharInfoLen < 4) {
1057f5736e95SDavid du Colombier 		DBG_DEC(tCharInfoLen);
1058f5736e95SDavid du Colombier 		return;
1059f5736e95SDavid du Colombier 	}
1060f5736e95SDavid du Colombier 
1061f5736e95SDavid du Colombier 	aucBuffer = xmalloc(tCharInfoLen);
1062f5736e95SDavid du Colombier 	if (!bReadBuffer(pFile, ulStartBlock,
1063f5736e95SDavid du Colombier 			aulBBD, tBBDLen, BIG_BLOCK_SIZE,
1064f5736e95SDavid du Colombier 			aucBuffer, ulBeginCharInfo, tCharInfoLen)) {
1065f5736e95SDavid du Colombier 		aucBuffer = xfree(aucBuffer);
1066f5736e95SDavid du Colombier 		return;
1067f5736e95SDavid du Colombier 	}
1068f5736e95SDavid du Colombier 
1069f5736e95SDavid du Colombier 	tLen = (tCharInfoLen - 4) / 6;
1070f5736e95SDavid du Colombier 	ausCharPage = xcalloc(tLen, sizeof(USHORT));
1071*25b329d5SDavid du Colombier 	for (tIndex = 0, tOffset = (tLen + 1) * 4;
1072*25b329d5SDavid du Colombier 	     tIndex < tLen;
1073*25b329d5SDavid du Colombier 	     tIndex++, tOffset += 2) {
1074*25b329d5SDavid du Colombier 		 ausCharPage[tIndex] = usGetWord(tOffset, aucBuffer);
1075*25b329d5SDavid du Colombier 		 NO_DBG_DEC(ausCharPage[tIndex]);
1076f5736e95SDavid du Colombier 	}
1077f5736e95SDavid du Colombier 	DBG_HEX(ulGetLong(0, aucBuffer));
1078f5736e95SDavid du Colombier 	aucBuffer = xfree(aucBuffer);
1079f5736e95SDavid du Colombier 	tCharPageNum = (size_t)usGetWord(0x18e, aucHeader); /* cpnBteChp */
1080f5736e95SDavid du Colombier 	DBG_DEC(tCharPageNum);
1081f5736e95SDavid du Colombier 	if (tLen < tCharPageNum) {
1082f5736e95SDavid du Colombier 		/* Replace CharPage by a longer version */
1083f5736e95SDavid du Colombier 		tLenOld = tLen;
1084f5736e95SDavid du Colombier 		usCharFirstPage = usGetWord(0x18a, aucHeader); /* pnChrFirst */
1085f5736e95SDavid du Colombier 		DBG_DEC(usCharFirstPage);
1086f5736e95SDavid du Colombier 		tLen += tCharPageNum - 1;
1087f5736e95SDavid du Colombier 		tSize = tLen * sizeof(USHORT);
1088f5736e95SDavid du Colombier 		ausCharPage = xrealloc(ausCharPage, tSize);
1089f5736e95SDavid du Colombier 		/* Add new values */
1090f5736e95SDavid du Colombier 		usCount = usCharFirstPage + 1;
1091*25b329d5SDavid du Colombier 		for (tIndex = tLenOld; tIndex < tLen; tIndex++) {
1092*25b329d5SDavid du Colombier 			ausCharPage[tIndex] = usCount;
1093*25b329d5SDavid du Colombier 			NO_DBG_DEC(ausCharPage[tIndex]);
1094f5736e95SDavid du Colombier 			usCount++;
1095f5736e95SDavid du Colombier 		}
1096f5736e95SDavid du Colombier 	}
1097f5736e95SDavid du Colombier 
1098*25b329d5SDavid du Colombier 	for (tIndex = 0; tIndex < tLen; tIndex++) {
1099f5736e95SDavid du Colombier 		if (!bReadBuffer(pFile, ulStartBlock,
1100f5736e95SDavid du Colombier 				aulBBD, tBBDLen, BIG_BLOCK_SIZE,
1101f5736e95SDavid du Colombier 				aucFpage,
1102*25b329d5SDavid du Colombier 				(ULONG)ausCharPage[tIndex] * BIG_BLOCK_SIZE,
1103f5736e95SDavid du Colombier 				BIG_BLOCK_SIZE)) {
1104f5736e95SDavid du Colombier 			break;
1105f5736e95SDavid du Colombier 		}
1106*25b329d5SDavid du Colombier 		tRun = (size_t)ucGetByte(0x1ff, aucFpage);
1107*25b329d5SDavid du Colombier 		NO_DBG_DEC(tRun);
1108*25b329d5SDavid du Colombier 		for (tIndex2 = 0; tIndex2 < tRun; tIndex2++) {
1109*25b329d5SDavid du Colombier 		  	ulCharPos = ulGetLong(tIndex2 * 4, aucFpage);
1110f5736e95SDavid du Colombier 			ulFileOffset = ulCharPos2FileOffset(ulCharPos);
1111f5736e95SDavid du Colombier 			iFodo = 2 * (int)ucGetByte(
1112*25b329d5SDavid du Colombier 				(tRun + 1) * 4 + tIndex2, aucFpage);
1113f5736e95SDavid du Colombier 
1114f5736e95SDavid du Colombier 			iLen = (int)ucGetByte(iFodo, aucFpage);
1115f5736e95SDavid du Colombier 
1116f5736e95SDavid du Colombier 			usIstd = usGetIstd(ulFileOffset);
1117f5736e95SDavid du Colombier 			vFillFontFromStylesheet(usIstd, &tFont);
1118f5736e95SDavid du Colombier 			if (iFodo != 0) {
1119f5736e95SDavid du Colombier 				vGet6FontInfo(iFodo, usIstd,
1120f5736e95SDavid du Colombier 					aucFpage + 1, iLen - 1, &tFont);
1121f5736e95SDavid du Colombier 			}
1122f5736e95SDavid du Colombier 			tFont.ulFileOffset = ulFileOffset;
1123f5736e95SDavid du Colombier 			vAdd2FontInfoList(&tFont);
1124f5736e95SDavid du Colombier 
1125f5736e95SDavid du Colombier 			if (iFodo <= 0) {
1126f5736e95SDavid du Colombier 				continue;
1127f5736e95SDavid du Colombier 			}
1128f5736e95SDavid du Colombier 
1129f5736e95SDavid du Colombier 			(void)memset(&tPicture, 0, sizeof(tPicture));
1130f5736e95SDavid du Colombier 			if (bGet6PicInfo(iFodo, aucFpage + 1,
1131f5736e95SDavid du Colombier 						iLen - 1, &tPicture)) {
1132f5736e95SDavid du Colombier 				tPicture.ulFileOffset = ulFileOffset;
1133f5736e95SDavid du Colombier 				tPicture.ulFileOffsetPicture =
1134f5736e95SDavid du Colombier 					ulDataPos2FileOffset(
1135f5736e95SDavid du Colombier 						tPicture.ulPictureOffset);
1136f5736e95SDavid du Colombier 				vAdd2PictInfoList(&tPicture);
1137f5736e95SDavid du Colombier 			}
1138f5736e95SDavid du Colombier 		}
1139f5736e95SDavid du Colombier 	}
1140f5736e95SDavid du Colombier 	ausCharPage = xfree(ausCharPage);
1141f5736e95SDavid du Colombier } /* end of vGet6ChrInfo */
1142