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