source: trunk/dll/literal.c@ 466

Last change on this file since 466 was 409, checked in by root, 19 years ago

Comments

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 6.6 KB
RevLine 
[117]1
2/***********************************************************************
3
4 $Id: literal.c 409 2006-07-29 20:08:43Z root $
5
6 string quoting utilities
7 wildcarding utilities
8
9 Copyright (c) 1993-98 M. Kimes
[298]10 Copyright (c) 2004, 2006 Steven H.Levine
[117]11
12 Archive containers
13
[298]14 01 Aug 04 SHL Rework fixup to avoid overflows
15 16 Jun 06 SHL liternal: comments
[351]16 22 Jul 06 SHL Check more run time errors
[117]17
18***********************************************************************/
19
[2]20#define INCL_OS2
21#define INCL_WIN
[351]22#include <os2.h>
[2]23
24#include <ctype.h>
25#include <stdlib.h>
26#include <stdio.h>
27#include <string.h>
[351]28
[2]29#include "fm3dll.h"
30
[351]31static PSZ pszSrcFile = __FILE__;
32
[117]33static INT index(const CHAR *s,const CHAR c);
34
[2]35#pragma alloc_text(LITERAL,literal,index,fixup,wildcard)
36
[117]37/* Get index of char in string
38 * @parm s string to search
39 * @parm c char to search for
40 * @return 0 relative index of c in s or -1
41 */
[2]42
[117]43static INT index(const CHAR *s,const CHAR c)
44{
[2]45 CHAR *p;
46
47 p = strchr(s,c);
48 if(p == NULL || !*p)
49 return -1;
50 return (INT)(p - s);
51}
52
[409]53/* literal()
54 * Translate a string with \ escape tokens to binary equivalent
[117]55 * Translates in place
56 *
[2]57 * 1. \x1b translates to CHAR(0x1b)
58 * 2. \27 translates to CHAR(27)
59 * 3. \" translates to "
60 * 4. \' translates to '
61 * 5. \\ translates to \
62 * 6. \r translates to carriage return
63 * 7. \n translates to linefeed
64 * 8. \b translates to backspace
65 * 9. \t translates to tab
66 * 10. \a translates to bell
67 * 11. \f translates to formfeed
68 *
69 * Synopsis
70 * *s = "this\x20is\32a test of \\MSC\\CSM\7"
71 * literal(s);
72 *
73 * ( s now equals "this is a test of \MSC\CSM")
[298]74 *
[409]75 * Return converted character count like strlen()
76 * Count does not include terminating nul
[2]77 */
78
79#define HEX "0123456789ABCDEF"
80#define DEC "0123456789"
81
[117]82UINT literal(PSZ pszBuf)
83{
84 INT wpos;
85 INT iBuf;
86 UINT cBufBytes;
87 INT iBufSave;
88 PSZ pszOut;
89 PSZ pszWork;
90 CHAR wchar;
[2]91
[117]92 if(!pszBuf || !*pszBuf)
[2]93 return 0;
[117]94 cBufBytes = strlen(pszBuf) + 1;
[351]95 pszWork = pszOut = xmalloc(cBufBytes + 1,pszSrcFile,__LINE__);
[2]96
[117]97 iBuf = 0; /* set index to first character */
98 while(pszBuf[iBuf]) {
99 switch(pszBuf[iBuf]) {
[2]100 case '\\':
[117]101 switch(pszBuf[iBuf + 1]) {
102 case 'x' : /* hexadecimal */
103 wchar = 0;
104 iBuf += 2; /* get past "\x" */
105 if(index(HEX,(CHAR)toupper(pszBuf[iBuf])) != -1) {
106 iBufSave = iBuf;
107 while(((wpos = index(HEX,(CHAR)toupper(pszBuf[iBuf]))) != -1) &&
108 iBuf < iBufSave + 2) {
109 wchar = (CHAR)(wchar << 4) + (CHAR)wpos;
110 iBuf++;
111 }
112 }
113 else
114 wchar = 'x'; /* just an x */
115 iBuf--;
116 *pszOut++ = wchar;
117 break;
[2]118
[117]119 case '\\' : /* we want a "\" */
120 iBuf++;
121 *pszOut++ = '\\';
122 break;
[2]123
[117]124 case 't' : /* tab CHAR */
125 iBuf++;
126 *pszOut++ = '\t';
127 break;
[2]128
[117]129 case 'n' : /* new line */
130 iBuf++;
131 *pszOut++ = '\n';
132 break;
[2]133
[117]134 case 'r' : /* carr return */
135 iBuf++;
136 *pszOut++ = '\r';
137 break;
[2]138
[117]139 case 'b' : /* back space */
140 iBuf++;
141 *pszOut++ = '\b';
142 break;
[2]143
[117]144 case 'f': /* formfeed */
145 iBuf++;
146 *pszOut++ = '\x0c';
147 break;
[2]148
[117]149 case 'a': /* bell */
150 iBuf++;
151 *pszOut++ = '\07';
152 break;
[2]153
[117]154 case '\'' : /* single quote */
155 iBuf++;
156 *pszOut++ = '\'';
157 break;
[2]158
[117]159 case '\"' : /* double quote */
160 iBuf++;
161 *pszOut++ = '\"';
162 break;
[2]163
[117]164 default : /* decimal */
165 iBuf++; /* get past "\" */
166 wchar = 0;
167 if(index(DEC,pszBuf[iBuf]) != -1) {
168 iBufSave = iBuf;
169 do { /* cvt to binary */
170 wchar = (CHAR)(wchar * 10 + (pszBuf[iBuf++] - 48));
171 } while (index(DEC,pszBuf[iBuf]) != -1 && iBuf < iBufSave + 3);
172 iBuf--;
173 }
174 else
175 wchar = pszBuf[iBuf];
176 *pszOut ++ = wchar;
177 break;
178 } // switch
179 break;
[2]180
181 default :
[117]182 *pszOut++ = pszBuf[iBuf];
183 break;
184 } // switch
185 iBuf++;
186 } // while
[298]187 *pszOut = 0; /* Always terminate, even if not string */
[2]188
[298]189 cBufBytes = pszOut - pszWork; /* Calc string length excluding terminator */
[117]190 memcpy(pszBuf,pszWork,cBufBytes + 1); /* Overwrite including terminator */
191 free(pszWork);
[2]192
[298]193 return cBufBytes; /* Return string length */
[2]194}
195
[117]196/* Check wildcard match
197 * @parm pszBuf Buffer to check
198 * @parm pszWildCard wildcard to match
199 * @parm fNotFileSpec TRUE if generic match else filespec match
200 * @return TRUE if matched else FALSE
201 */
[2]202
203
[117]204BOOL wildcard(const PSZ pszBuf,const PSZ pszWildCard,const BOOL fNotFileSpec)
205{
206 const CHAR *fstr = pszBuf;
207 PSZ fcard = pszWildCard;
208 INT wmatch = TRUE;
[2]209
210 while(wmatch && *fcard && *fstr) {
211 switch(*fcard) {
212 case '?' : /* character substitution */
[117]213 fcard++;
214 if(fNotFileSpec || (*fstr != '.' && *fstr != '/' && *fstr != '\\'))
215 fstr++; /* skip (match) next character */
216 break;
[2]217
218 case '*' :
[117]219 /* find next non-wild character in wildcard */
220 while(*fcard && (*fcard == '?' || *fcard == '*'))
221 fcard++;
222 if(!*fcard) /* if last char of wildcard is *, it matches */
223 return TRUE;
224 /* skip until partition, match, or eos */
225 while(*fstr && toupper(*fstr) != toupper(*fcard) &&
226 (fNotFileSpec || (*fstr != '\\' &&
227 *fstr != '/' && *fstr != '.')))
228 fstr++;
229 if(!fNotFileSpec && !*fstr) /* implicit '.' */
230 if(*fcard == '.')
231 fcard++;
232 break;
[2]233
234 default :
[117]235 if(!fNotFileSpec && ((*fstr == '/' || *fstr == '\\') &&
236 (*fcard == '/' || *fcard == '\\')))
237 wmatch = TRUE;
238 else
239 wmatch = (toupper(*fstr) == toupper(*fcard));
240 fstr++;
241 fcard++;
242 break;
[2]243 }
244 }
245
246 if ((*fcard && *fcard != '*') || *fstr)
247 return 0;
248 else
249 return wmatch;
250}
251
252
[117]253// fixup - quote literal character array
[2]254
[117]255PSZ fixup(const PCH pachIn, PSZ pszOutBuf, const UINT cBufBytes, const UINT cInBytes)
256{
257 PCH pchIn = pachIn;
258 PCH pchOut = pszOutBuf;
259 PSZ pszTemp;
260 static CHAR szTemp[5] = "\\x"; // Constant prefix
[2]261
[117]262 // input is a character array, not a string - may not be null terminated
263 // cBufBytes is buffer size
264 if (pachIn) {
265 // Ensure room for null and possible \ escape
266 while (pchIn - pachIn < cInBytes &&
267 pchOut - pszOutBuf + 2 < cBufBytes) {
268 if(!isprint(*pchIn)) {
269 if(*pchIn == '\r') {
270 *pchOut++ = '\\';
271 *pchOut++ = 'r';
272 }
273 else if(*pchIn == '\n') {
274 *pchOut++ = '\\';
275 *pchOut++ = 'n';
276 }
277 else if(*pchIn == '\b') {
278 *pchOut++ = '\\';
279 *pchOut++ = 'b';
280 }
281 else {
282 sprintf(szTemp + 2,"%02hx",*pchIn);
283 for (pszTemp = szTemp; *pszTemp;)
284 *pchOut++ = *pszTemp++;
285 }
286 pchIn++;
[2]287 }
[117]288 else if(*pchIn == '\\') {
289 *pchOut++ = '\\';
290 *pchOut++ = '\\';
291 pchIn++;
[2]292 }
[117]293 else
294 *pchOut++ = *pchIn++;
295 } // while
296 } // if pachIn
297 *pchOut = 0;
298 return pszOutBuf;
[2]299}
300
Note: See TracBrowser for help on using the repository browser.