1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59#include <sys/cdefs.h>
60
61#if 0
62#include <ctype.h>
63#else
64static inline int
65isspace(char c)
66{
67 return (c == ' ' || c == '\t' || c == '\n' || c == '\12');
68}
69#endif
70#include <stdarg.h>
71#include <string.h>
72#include <sys/param.h>
73#include <sys/systm.h>
74
75#define BUF 32
76
77
78
79
80#define LONG 0x01
81#define SHORT 0x04
82#define SUPPRESS 0x08
83#define POINTER 0x10
84#define NOSKIP 0x20
85#define LONGLONG 0x400
86#define SHORTSHORT 0x4000
87#define UNSIGNED 0x8000
88
89
90
91
92
93
94#define SIGNOK 0x40
95#define NDIGITS 0x80
96
97#define DPTOK 0x100
98#define EXPOK 0x200
99
100#define PFXOK 0x100
101#define NZDIGITS 0x200
102
103
104
105
106#define CT_CHAR 0
107#define CT_CCL 1
108#define CT_STRING 2
109#define CT_INT 3
110
111static const u_char *__sccl(char *, const u_char *);
112
113int
114sscanf(const char *ibuf, const char *fmt, ...)
115{
116 va_list ap;
117 int ret;
118
119 va_start(ap, fmt);
120 ret = vsscanf(ibuf, fmt, ap);
121 va_end(ap);
122 return(ret);
123}
124
125int
126vsscanf(const char *inp, char const *fmt0, va_list ap)
127{
128 int inr;
129 const u_char *fmt = (const u_char *)fmt0;
130 int c;
131 size_t width;
132 char *p;
133 int n;
134 int flags;
135 char *p0;
136 int nassigned;
137 int nconversions;
138 int nread;
139 int base;
140 char ccltab[256];
141 char buf[BUF];
142
143
144 static short basefix[17] =
145 { 10, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 };
146
147 inr = strlen(inp);
148
149 nassigned = 0;
150 nconversions = 0;
151 nread = 0;
152 base = 0;
153 for (;;) {
154 c = *fmt++;
155 if (c == 0)
156 return (nassigned);
157 if (isspace(c)) {
158 while (inr > 0 && isspace(*inp))
159 nread++, inr--, inp++;
160 continue;
161 }
162 if (c != '%')
163 goto literal;
164 width = 0;
165 flags = 0;
166
167
168
169
170again: c = *fmt++;
171 switch (c) {
172 case '%':
173literal:
174 if (inr <= 0)
175 goto input_failure;
176 if (*inp != c)
177 goto match_failure;
178 inr--, inp++;
179 nread++;
180 continue;
181
182 case '*':
183 flags |= SUPPRESS;
184 goto again;
185 case 'l':
186 if (flags & LONG) {
187 flags &= ~LONG;
188 flags |= LONGLONG;
189 } else
190 flags |= LONG;
191 goto again;
192 case 'q':
193 flags |= LONGLONG;
194 goto again;
195 case 'h':
196 if (flags & SHORT) {
197 flags &= ~SHORT;
198 flags |= SHORTSHORT;
199 } else
200 flags |= SHORT;
201 goto again;
202
203 case '0': case '1': case '2': case '3': case '4':
204 case '5': case '6': case '7': case '8': case '9':
205 width = width * 10 + c - '0';
206 goto again;
207
208
209
210
211 case 'd':
212 c = CT_INT;
213 base = 10;
214 break;
215
216 case 'i':
217 c = CT_INT;
218 base = 0;
219 break;
220
221 case 'o':
222 c = CT_INT;
223 flags |= UNSIGNED;
224 base = 8;
225 break;
226
227 case 'u':
228 c = CT_INT;
229 flags |= UNSIGNED;
230 base = 10;
231 break;
232
233 case 'X':
234 case 'x':
235 flags |= PFXOK;
236 c = CT_INT;
237 flags |= UNSIGNED;
238 base = 16;
239 break;
240
241 case 's':
242 c = CT_STRING;
243 break;
244
245 case '[':
246 fmt = __sccl(ccltab, fmt);
247 flags |= NOSKIP;
248 c = CT_CCL;
249 break;
250
251 case 'c':
252 flags |= NOSKIP;
253 c = CT_CHAR;
254 break;
255
256 case 'p':
257 flags |= POINTER | PFXOK;
258 c = CT_INT;
259 flags |= UNSIGNED;
260 base = 16;
261 break;
262
263 case 'n':
264 nconversions++;
265 if (flags & SUPPRESS)
266 continue;
267 if (flags & SHORTSHORT)
268 *va_arg(ap, char *) = nread;
269 else if (flags & SHORT)
270 *va_arg(ap, short *) = nread;
271 else if (flags & LONG)
272 *va_arg(ap, long *) = nread;
273 else if (flags & LONGLONG)
274 *va_arg(ap, long long *) = nread;
275 else
276 *va_arg(ap, int *) = nread;
277 continue;
278 }
279
280
281
282
283 if (inr <= 0)
284 goto input_failure;
285
286
287
288
289
290 if ((flags & NOSKIP) == 0) {
291 while (isspace(*inp)) {
292 nread++;
293 if (--inr > 0)
294 inp++;
295 else
296 goto input_failure;
297 }
298
299
300
301
302
303 }
304
305
306
307
308 switch (c) {
309
310 case CT_CHAR:
311
312 if (width == 0)
313 width = 1;
314 if (flags & SUPPRESS) {
315 size_t sum = 0;
316 for (;;) {
317 if ((n = inr) < (int)width) {
318 sum += n;
319 width -= n;
320 inp += n;
321 if (sum == 0)
322 goto input_failure;
323 break;
324 } else {
325 sum += width;
326 inr -= width;
327 inp += width;
328 break;
329 }
330 }
331 nread += sum;
332 } else {
333 bcopy(inp, va_arg(ap, char *), width);
334 inr -= width;
335 inp += width;
336 nread += width;
337 nassigned++;
338 }
339 nconversions++;
340 break;
341
342 case CT_CCL:
343
344 if (width == 0)
345 width = (size_t)~0;
346
347 if (flags & SUPPRESS) {
348 n = 0;
349 while (ccltab[(unsigned char)*inp]) {
350 n++, inr--, inp++;
351 if (--width == 0)
352 break;
353 if (inr <= 0) {
354 if (n == 0)
355 goto input_failure;
356 break;
357 }
358 }
359 if (n == 0)
360 goto match_failure;
361 } else {
362 p0 = p = va_arg(ap, char *);
363 while (ccltab[(unsigned char)*inp]) {
364 inr--;
365 *p++ = *inp++;
366 if (--width == 0)
367 break;
368 if (inr <= 0) {
369 if (p == p0)
370 goto input_failure;
371 break;
372 }
373 }
374 n = p - p0;
375 if (n == 0)
376 goto match_failure;
377 *p = 0;
378 nassigned++;
379 }
380 nread += n;
381 nconversions++;
382 break;
383
384 case CT_STRING:
385
386 if (width == 0)
387 width = (size_t)~0;
388 if (flags & SUPPRESS) {
389 n = 0;
390 while (!isspace(*inp)) {
391 n++, inr--, inp++;
392 if (--width == 0)
393 break;
394 if (inr <= 0)
395 break;
396 }
397 nread += n;
398 } else {
399 p0 = p = va_arg(ap, char *);
400 while (!isspace(*inp)) {
401 inr--;
402 *p++ = *inp++;
403 if (--width == 0)
404 break;
405 if (inr <= 0)
406 break;
407 }
408 *p = 0;
409 nread += p - p0;
410 nassigned++;
411 }
412 nconversions++;
413 continue;
414
415 case CT_INT:
416
417#ifdef hardway
418 if (width == 0 || width > sizeof(buf) - 1)
419 width = sizeof(buf) - 1;
420#else
421
422 if (--width > sizeof(buf) - 2)
423 width = sizeof(buf) - 2;
424 width++;
425#endif
426 flags |= SIGNOK | NDIGITS | NZDIGITS;
427 for (p = buf; width; width--) {
428 c = *inp;
429
430
431
432
433 switch (c) {
434
435
436
437
438
439
440
441
442
443
444
445
446
447 case '0':
448 if (base == 0) {
449 base = 8;
450 flags |= PFXOK;
451 }
452 if (flags & NZDIGITS)
453 flags &= ~(SIGNOK|NZDIGITS|NDIGITS);
454 else
455 flags &= ~(SIGNOK|PFXOK|NDIGITS);
456 goto ok;
457
458
459 case '1': case '2': case '3':
460 case '4': case '5': case '6': case '7':
461 base = basefix[base];
462 flags &= ~(SIGNOK | PFXOK | NDIGITS);
463 goto ok;
464
465
466 case '8': case '9':
467 base = basefix[base];
468 if (base <= 8)
469 break;
470 flags &= ~(SIGNOK | PFXOK | NDIGITS);
471 goto ok;
472
473
474 case 'A': case 'B': case 'C':
475 case 'D': case 'E': case 'F':
476 case 'a': case 'b': case 'c':
477 case 'd': case 'e': case 'f':
478
479 if (base <= 10)
480 break;
481 flags &= ~(SIGNOK | PFXOK | NDIGITS);
482 goto ok;
483
484
485 case '+': case '-':
486 if (flags & SIGNOK) {
487 flags &= ~SIGNOK;
488 goto ok;
489 }
490 break;
491
492
493 case 'x': case 'X':
494 if (flags & PFXOK && p == buf + 1) {
495 base = 16;
496 flags &= ~PFXOK;
497 goto ok;
498 }
499 break;
500 }
501
502
503
504
505
506 break;
507 ok:
508
509
510
511 *p++ = c;
512 if (--inr > 0)
513 inp++;
514 else
515 break;
516 }
517
518
519
520
521
522
523 if (flags & NDIGITS) {
524 if (p > buf) {
525 inp--;
526 inr++;
527 }
528 goto match_failure;
529 }
530 c = ((u_char *)p)[-1];
531 if (c == 'x' || c == 'X') {
532 --p;
533 inp--;
534 inr++;
535 }
536 if ((flags & SUPPRESS) == 0) {
537 u_quad_t res;
538
539 *p = 0;
540 if ((flags & UNSIGNED) == 0)
541 res = strtoq(buf, (char **)NULL, base);
542 else
543 res = strtouq(buf, (char **)NULL, base);
544 if (flags & POINTER)
545 *va_arg(ap, void **) =
546 (void *)(uintptr_t)res;
547 else if (flags & SHORTSHORT)
548 *va_arg(ap, char *) = res;
549 else if (flags & SHORT)
550 *va_arg(ap, short *) = res;
551 else if (flags & LONG)
552 *va_arg(ap, long *) = res;
553 else if (flags & LONGLONG)
554 *va_arg(ap, long long *) = res;
555 else
556 *va_arg(ap, int *) = res;
557 nassigned++;
558 }
559 nread += p - buf;
560 nconversions++;
561 break;
562
563 }
564 }
565input_failure:
566 return (nconversions != 0 ? nassigned : -1);
567match_failure:
568 return (nassigned);
569}
570
571
572
573
574
575
576
577static const u_char *
578__sccl(char *tab, const u_char *fmt)
579{
580 int c, n, v;
581
582
583 c = *fmt++;
584 if (c == '^') {
585 v = 1;
586 c = *fmt++;
587 } else
588 v = 0;
589
590
591 (void) memset(tab, v, 256);
592
593 if (c == 0)
594 return (fmt - 1);
595
596
597
598
599
600
601
602
603 v = 1 - v;
604 for (;;) {
605 tab[c] = v;
606doswitch:
607 n = *fmt++;
608 switch (n) {
609
610 case 0:
611 return (fmt - 1);
612
613 case '-':
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632 n = *fmt;
633 if (n == ']' || n < c) {
634 c = '-';
635 break;
636 }
637 fmt++;
638
639 do {
640 tab[++c] = v;
641 } while (c < n);
642 c = n;
643
644
645
646
647
648 goto doswitch;
649 break;
650
651 case ']':
652 return (fmt);
653
654 default:
655 c = n;
656 break;
657 }
658 }
659
660}
661