1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /*
4 *******************************************************************************
5 *
6 *   Copyright (C) 1999-2010, International Business Machines
7 *   Corporation and others.  All Rights Reserved.
8 *
9 *******************************************************************************
10 *   file name:  unewdata.c
11 *   encoding:   UTF-8
12 *   tab size:   8 (not used)
13 *   indentation:4
14 *
15 *   created on: 1999oct25
16 *   created by: Markus W. Scherer
17 */
18 
19 #include <stdio.h>
20 #include "unicode/utypes.h"
21 #include "unicode/putil.h"
22 #include "unicode/ustring.h"
23 #include "cmemory.h"
24 #include "cstring.h"
25 #include "filestrm.h"
26 #include "unicode/udata.h"
27 #include "unewdata.h"
28 
29 struct UNewDataMemory {
30     FileStream *file;
31     uint16_t headerSize;
32     uint8_t magic1, magic2;
33 };
34 
35 U_CAPI UNewDataMemory * U_EXPORT2
udata_create(const char * dir,const char * type,const char * name,const UDataInfo * pInfo,const char * comment,UErrorCode * pErrorCode)36 udata_create(const char *dir, const char *type, const char *name,
37              const UDataInfo *pInfo,
38              const char *comment,
39              UErrorCode *pErrorCode) {
40     UNewDataMemory *pData;
41     uint16_t headerSize, commentLength;
42     char filename[512];
43     uint8_t bytes[16];
44     int32_t length;
45 
46     if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
47         return NULL;
48     } else if(name==NULL || *name==0 || pInfo==NULL) {
49         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
50         return NULL;
51     }
52 
53     /* allocate the data structure */
54     pData=(UNewDataMemory *)uprv_malloc(sizeof(UNewDataMemory));
55     if(pData==NULL) {
56         *pErrorCode=U_MEMORY_ALLOCATION_ERROR;
57         return NULL;
58     }
59 
60     char dirSepChar = U_FILE_SEP_CHAR;
61 #if (U_FILE_SEP_CHAR != U_FILE_ALT_SEP_CHAR)
62     // We may need to append a different directory separator when building for Cygwin or MSYS2.
63     if(dir && *dir) {
64       if(!uprv_strchr(dir, U_FILE_SEP_CHAR) && uprv_strchr(dir, U_FILE_ALT_SEP_CHAR)) {
65           dirSepChar = U_FILE_ALT_SEP_CHAR;
66       }
67     }
68 #endif
69 
70     /* Check that the full path won't be too long */
71     length = 0;					/* Start with nothing */
72     if(dir != NULL  && *dir !=0)	/* Add directory length if one was given */
73     {
74     	length += static_cast<int32_t>(strlen(dir));
75 
76     	/* Add 1 if dir doesn't end with path sep */
77         if (dir[strlen(dir) - 1]!= dirSepChar) {
78             length++;
79         }
80 	}
81     length += static_cast<int32_t>(strlen(name));		/* Add the filename length */
82 
83     if(type != NULL  && *type !=0) { /* Add directory length if  given */
84         length += static_cast<int32_t>(strlen(type));
85     }
86 
87 
88      /* LDH buffer Length error check */
89     if(length  > ((int32_t)sizeof(filename) - 1))
90     {
91    	    *pErrorCode = U_BUFFER_OVERFLOW_ERROR;
92    	    uprv_free(pData);
93 	    return NULL;
94     }
95 
96     /* open the output file */
97     if(dir!=NULL && *dir!=0) { /* if dir has a value, we prepend it to the filename */
98         char *p=filename+strlen(dir);
99         uprv_strcpy(filename, dir);
100         if (*(p-1)!=dirSepChar) {
101             *p++=dirSepChar;
102             *p=0;
103         }
104     } else { /* otherwise, we'll output to the current dir */
105         filename[0]=0;
106     }
107     uprv_strcat(filename, name);
108     if(type!=NULL && *type!=0) {
109         uprv_strcat(filename, ".");
110         uprv_strcat(filename, type);
111     }
112     pData->file=T_FileStream_open(filename, "wb");
113     if(pData->file==NULL) {
114         uprv_free(pData);
115         *pErrorCode=U_FILE_ACCESS_ERROR;
116         return NULL;
117     }
118 
119     /* write the header information */
120     headerSize=(uint16_t)(pInfo->size+4);
121     if(comment!=NULL && *comment!=0) {
122         commentLength=(uint16_t)(uprv_strlen(comment)+1);
123         headerSize+=commentLength;
124     } else {
125         commentLength=0;
126     }
127 
128     /* write the size of the header, take padding into account */
129     pData->headerSize=(uint16_t)((headerSize+15)&~0xf);
130     pData->magic1=0xda;
131     pData->magic2=0x27;
132     T_FileStream_write(pData->file, &pData->headerSize, 4);
133 
134     /* write the information data */
135     T_FileStream_write(pData->file, pInfo, pInfo->size);
136 
137     /* write the comment */
138     if(commentLength>0) {
139         T_FileStream_write(pData->file, comment, commentLength);
140     }
141 
142     /* write padding bytes to align the data section to 16 bytes */
143     headerSize&=0xf;
144     if(headerSize!=0) {
145         headerSize=(uint16_t)(16-headerSize);
146         uprv_memset(bytes, 0, headerSize);
147         T_FileStream_write(pData->file, bytes, headerSize);
148     }
149 
150     return pData;
151 }
152 
153 U_CAPI uint32_t U_EXPORT2
udata_finish(UNewDataMemory * pData,UErrorCode * pErrorCode)154 udata_finish(UNewDataMemory *pData, UErrorCode *pErrorCode) {
155     uint32_t fileLength=0;
156 
157     if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
158         return 0;
159     }
160 
161     if(pData!=NULL) {
162         if(pData->file!=NULL) {
163             /* fflush(pData->file);*/
164             fileLength=T_FileStream_size(pData->file);
165             if(T_FileStream_error(pData->file)) {
166                 *pErrorCode=U_FILE_ACCESS_ERROR;
167             } else {
168                 fileLength-=pData->headerSize;
169             }
170             T_FileStream_close(pData->file);
171         }
172         uprv_free(pData);
173     }
174 
175     return fileLength;
176 }
177 
178 /* dummy UDataInfo cf. udata.h */
179 static const UDataInfo dummyDataInfo = {
180     sizeof(UDataInfo),
181     0,
182 
183     U_IS_BIG_ENDIAN,
184     U_CHARSET_FAMILY,
185     U_SIZEOF_UCHAR,
186     0,
187 
188     { 0, 0, 0, 0 },                 /* dummy dataFormat */
189     { 0, 0, 0, 0 },                 /* dummy formatVersion */
190     { 0, 0, 0, 0 }                  /* dummy dataVersion */
191 };
192 
193 U_CAPI void U_EXPORT2
udata_createDummy(const char * dir,const char * type,const char * name,UErrorCode * pErrorCode)194 udata_createDummy(const char *dir, const char *type, const char *name, UErrorCode *pErrorCode) {
195     if(U_SUCCESS(*pErrorCode)) {
196         udata_finish(udata_create(dir, type, name, &dummyDataInfo, NULL, pErrorCode), pErrorCode);
197         if(U_FAILURE(*pErrorCode)) {
198             fprintf(stderr, "error %s writing dummy data file %s" U_FILE_SEP_STRING "%s.%s\n",
199                     u_errorName(*pErrorCode), dir, name, type);
200             exit(*pErrorCode);
201         }
202     }
203 }
204 
205 U_CAPI void U_EXPORT2
udata_write8(UNewDataMemory * pData,uint8_t byte)206 udata_write8(UNewDataMemory *pData, uint8_t byte) {
207     if(pData!=NULL && pData->file!=NULL) {
208         T_FileStream_write(pData->file, &byte, 1);
209     }
210 }
211 
212 U_CAPI void U_EXPORT2
udata_write16(UNewDataMemory * pData,uint16_t word)213 udata_write16(UNewDataMemory *pData, uint16_t word) {
214     if(pData!=NULL && pData->file!=NULL) {
215         T_FileStream_write(pData->file, &word, 2);
216     }
217 }
218 
219 U_CAPI void U_EXPORT2
udata_write32(UNewDataMemory * pData,uint32_t wyde)220 udata_write32(UNewDataMemory *pData, uint32_t wyde) {
221     if(pData!=NULL && pData->file!=NULL) {
222         T_FileStream_write(pData->file, &wyde, 4);
223     }
224 }
225 
226 U_CAPI void U_EXPORT2
udata_writeBlock(UNewDataMemory * pData,const void * s,int32_t length)227 udata_writeBlock(UNewDataMemory *pData, const void *s, int32_t length) {
228     if(pData!=NULL && pData->file!=NULL) {
229         if(length>0) {
230             T_FileStream_write(pData->file, s, length);
231         }
232     }
233 }
234 
235 U_CAPI void U_EXPORT2
udata_writePadding(UNewDataMemory * pData,int32_t length)236 udata_writePadding(UNewDataMemory *pData, int32_t length) {
237     static const uint8_t padding[16]={
238         0xaa, 0xaa, 0xaa, 0xaa,
239         0xaa, 0xaa, 0xaa, 0xaa,
240         0xaa, 0xaa, 0xaa, 0xaa,
241         0xaa, 0xaa, 0xaa, 0xaa
242     };
243     if(pData!=NULL && pData->file!=NULL) {
244         while(length>=16) {
245             T_FileStream_write(pData->file, padding, 16);
246             length-=16;
247         }
248         if(length>0) {
249             T_FileStream_write(pData->file, padding, length);
250         }
251     }
252 }
253 
254 U_CAPI void U_EXPORT2
udata_writeString(UNewDataMemory * pData,const char * s,int32_t length)255 udata_writeString(UNewDataMemory *pData, const char *s, int32_t length) {
256     if(pData!=NULL && pData->file!=NULL) {
257         if(length==-1) {
258             length=(int32_t)uprv_strlen(s);
259         }
260         if(length>0) {
261             T_FileStream_write(pData->file, s, length);
262         }
263     }
264 }
265 
266 U_CAPI void U_EXPORT2
udata_writeUString(UNewDataMemory * pData,const UChar * s,int32_t length)267 udata_writeUString(UNewDataMemory *pData, const UChar *s, int32_t length) {
268     if(pData!=NULL && pData->file!=NULL) {
269         if(length==-1) {
270             length=u_strlen(s);
271         }
272         if(length>0) {
273             T_FileStream_write(pData->file, s, length*sizeof(UChar));
274         }
275     }
276 }
277 
278 /*
279  * Hey, Emacs, please set the following:
280  *
281  * Local Variables:
282  * indent-tabs-mode: nil
283  * End:
284  *
285  */
286 
287