fmt.c 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569
  1. /*
  2. * Copyright (c) 2010-2018 Richard Braun.
  3. *
  4. * This program is free software: you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation, either version 3 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. *
  17. * Upstream site with license notes :
  18. * http://git.sceen.net/rbraun/librbraun.git/
  19. */
  20. #include <assert.h>
  21. #include <errno.h>
  22. #include <limits.h>
  23. #include <stdbool.h>
  24. #include <stdarg.h>
  25. #include <stddef.h>
  26. #include <stdint.h>
  27. #include <stdio.h>
  28. #include <string.h>
  29. #include <kern/fmt.h>
  30. #include <kern/macros.h>
  31. #include <kern/types.h>
  32. /*
  33. * Size for the temporary number buffer. The minimum base is 8 so 3 bits
  34. * are consumed per digit. Add one to round up. The conversion algorithm
  35. * doesn't use the null byte.
  36. */
  37. #define FMT_MAX_NUM_SIZE (((sizeof(unsigned long long) * CHAR_BIT) / 3) + 1)
  38. /*
  39. * Special size for fmt_vsnprintf(), used when the buffer size is unknown.
  40. */
  41. #define FMT_NOLIMIT ((size_t)-1)
  42. /*
  43. * Special character used to denote that the putback buffer is empty.
  44. */
  45. #define FMT_NOPUTBACK (EOF - 1)
  46. /*
  47. * Formatting flags.
  48. *
  49. * FMT_FORMAT_LOWER must be 0x20 as it is OR'd with fmt_digits, eg.
  50. * '0': 0x30 | 0x20 => 0x30 ('0')
  51. * 'A': 0x41 | 0x20 => 0x61 ('a')
  52. */
  53. #define FMT_FORMAT_ALT_FORM 0x0001 /* "Alternate form" */
  54. #define FMT_FORMAT_ZERO_PAD 0x0002 /* Zero padding on the left */
  55. #define FMT_FORMAT_LEFT_JUSTIFY 0x0004 /* Align text on the left */
  56. #define FMT_FORMAT_BLANK 0x0008 /* Blank space before positive number */
  57. #define FMT_FORMAT_SIGN 0x0010 /* Always place a sign (either + or -) */
  58. #define FMT_FORMAT_LOWER 0x0020 /* To lowercase (for %x) */
  59. #define FMT_FORMAT_CONV_SIGNED 0x0040 /* Format specifies signed conversion */
  60. #define FMT_FORMAT_DISCARD 0x0080 /* Discard output (scanf) */
  61. #define FMT_FORMAT_CHECK_WIDTH 0x0100 /* Check field width (scanf) */
  62. enum {
  63. FMT_MODIFIER_NONE,
  64. FMT_MODIFIER_CHAR,
  65. FMT_MODIFIER_SHORT,
  66. FMT_MODIFIER_LONG,
  67. FMT_MODIFIER_LONGLONG,
  68. FMT_MODIFIER_PTR, /* Used only for %p */
  69. FMT_MODIFIER_SIZE,
  70. FMT_MODIFIER_PTRDIFF,
  71. };
  72. enum {
  73. FMT_SPECIFIER_INVALID,
  74. FMT_SPECIFIER_INT,
  75. FMT_SPECIFIER_CHAR,
  76. FMT_SPECIFIER_STR,
  77. FMT_SPECIFIER_NRCHARS,
  78. FMT_SPECIFIER_PERCENT,
  79. };
  80. /*
  81. * Note that copies of the original va_list object are made, because va_arg()
  82. * may not reliably be used by different callee functions, and despite the
  83. * standard explicitely allowing pointers to va_list objects, it's apparently
  84. * very difficult for implementations to provide and is best avoided.
  85. */
  86. struct fmt_xprintf_state {
  87. const char *format;
  88. va_list ap;
  89. unsigned int flags;
  90. int width;
  91. int precision;
  92. unsigned int modifier;
  93. unsigned int specifier;
  94. unsigned int base;
  95. struct fmt_write_op write_op;
  96. int nr_chars;
  97. };
  98. struct fmt_xscanf_state {
  99. const char *format;
  100. va_list ap;
  101. unsigned int flags;
  102. int width;
  103. unsigned int modifier;
  104. unsigned int specifier;
  105. unsigned int base;
  106. int nr_convs;
  107. struct fmt_read_op read_op;
  108. int nr_chars;
  109. int cur_ch;
  110. };
  111. static const char fmt_digits[] = "0123456789ABCDEF";
  112. static char
  113. fmt_consume(const char **strp)
  114. {
  115. char c;
  116. c = **strp;
  117. (*strp)++;
  118. return c;
  119. }
  120. static bool
  121. fmt_isdigit(char c)
  122. {
  123. return (c >= '0') && (c <= '9');
  124. }
  125. static bool
  126. fmt_isxdigit(char c)
  127. {
  128. return fmt_isdigit(c)
  129. || ((c >= 'a') && (c <= 'f'))
  130. || ((c >= 'A') && (c <= 'F'));
  131. }
  132. static void
  133. fmt_xprintf_state_init(struct fmt_xprintf_state *state,
  134. struct fmt_write_op *op,
  135. const char *format, va_list ap)
  136. {
  137. state->format = format;
  138. va_copy(state->ap, ap);
  139. state->write_op = *op;
  140. state->nr_chars = 0;
  141. }
  142. static void
  143. fmt_xprintf_state_produce_raw_char(struct fmt_xprintf_state *state, char c)
  144. {
  145. state->write_op.putc(state->write_op.data, c);
  146. ++state->nr_chars;
  147. }
  148. static int
  149. fmt_xprintf_state_finalize(struct fmt_xprintf_state *state)
  150. {
  151. va_end(state->ap);
  152. return state->nr_chars;
  153. }
  154. static char
  155. fmt_xprintf_state_consume_format(struct fmt_xprintf_state *state)
  156. {
  157. return *state->format++;
  158. }
  159. static void
  160. fmt_xprintf_state_restore_format(struct fmt_xprintf_state *state)
  161. {
  162. --state->format;
  163. }
  164. static void
  165. fmt_xprintf_state_consume_flags(struct fmt_xprintf_state *state)
  166. {
  167. bool found;
  168. char c;
  169. found = true;
  170. state->flags = 0;
  171. do {
  172. c = fmt_xprintf_state_consume_format(state);
  173. switch (c) {
  174. case '#':
  175. state->flags |= FMT_FORMAT_ALT_FORM;
  176. break;
  177. case '0':
  178. state->flags |= FMT_FORMAT_ZERO_PAD;
  179. break;
  180. case '-':
  181. state->flags |= FMT_FORMAT_LEFT_JUSTIFY;
  182. break;
  183. case ' ':
  184. state->flags |= FMT_FORMAT_BLANK;
  185. break;
  186. case '+':
  187. state->flags |= FMT_FORMAT_SIGN;
  188. break;
  189. default:
  190. found = false;
  191. break;
  192. }
  193. } while (found);
  194. fmt_xprintf_state_restore_format(state);
  195. }
  196. static void
  197. fmt_xprintf_state_consume_width(struct fmt_xprintf_state *state)
  198. {
  199. char c;
  200. c = fmt_xprintf_state_consume_format(state);
  201. if (fmt_isdigit(c)) {
  202. state->width = 0;
  203. do {
  204. state->width = state->width * 10 + (c - '0');
  205. c = fmt_xprintf_state_consume_format(state);
  206. } while (fmt_isdigit(c));
  207. fmt_xprintf_state_restore_format(state);
  208. } else if (c == '*') {
  209. state->width = va_arg(state->ap, int);
  210. if (state->width < 0) {
  211. state->flags |= FMT_FORMAT_LEFT_JUSTIFY;
  212. state->width = -state->width;
  213. }
  214. } else {
  215. state->width = 0;
  216. fmt_xprintf_state_restore_format(state);
  217. }
  218. }
  219. static void
  220. fmt_xprintf_state_consume_precision(struct fmt_xprintf_state *state)
  221. {
  222. char c;
  223. c = fmt_xprintf_state_consume_format(state);
  224. if (c == '.') {
  225. c = fmt_xprintf_state_consume_format(state);
  226. if (fmt_isdigit(c)) {
  227. state->precision = 0;
  228. do {
  229. state->precision = state->precision * 10 + (c - '0');
  230. c = fmt_xprintf_state_consume_format(state);
  231. } while (fmt_isdigit(c));
  232. fmt_xprintf_state_restore_format(state);
  233. } else if (c == '*') {
  234. state->precision = va_arg(state->ap, int);
  235. if (state->precision < 0) {
  236. state->precision = 0;
  237. }
  238. } else {
  239. state->precision = 0;
  240. fmt_xprintf_state_restore_format(state);
  241. }
  242. } else {
  243. /* precision is >= 0 only if explicit */
  244. state->precision = -1;
  245. fmt_xprintf_state_restore_format(state);
  246. }
  247. }
  248. static void
  249. fmt_xprintf_state_consume_modifier(struct fmt_xprintf_state *state)
  250. {
  251. char c, c2;
  252. c = fmt_xprintf_state_consume_format(state);
  253. switch (c) {
  254. case 'h':
  255. case 'l':
  256. c2 = fmt_xprintf_state_consume_format(state);
  257. if (c == c2) {
  258. state->modifier = (c == 'h') ? FMT_MODIFIER_CHAR
  259. : FMT_MODIFIER_LONGLONG;
  260. } else {
  261. state->modifier = (c == 'h') ? FMT_MODIFIER_SHORT
  262. : FMT_MODIFIER_LONG;
  263. fmt_xprintf_state_restore_format(state);
  264. }
  265. break;
  266. case 'z':
  267. state->modifier = FMT_MODIFIER_SIZE;
  268. __fallthrough;
  269. case 't':
  270. state->modifier = FMT_MODIFIER_PTRDIFF;
  271. break;
  272. default:
  273. state->modifier = FMT_MODIFIER_NONE;
  274. fmt_xprintf_state_restore_format(state);
  275. break;
  276. }
  277. }
  278. static void
  279. fmt_xprintf_state_consume_specifier(struct fmt_xprintf_state *state)
  280. {
  281. char c;
  282. c = fmt_xprintf_state_consume_format(state);
  283. switch (c) {
  284. case 'd':
  285. case 'i':
  286. state->flags |= FMT_FORMAT_CONV_SIGNED;
  287. __fallthrough;
  288. case 'u':
  289. state->base = 10;
  290. state->specifier = FMT_SPECIFIER_INT;
  291. break;
  292. case 'o':
  293. state->base = 8;
  294. state->specifier = FMT_SPECIFIER_INT;
  295. break;
  296. case 'p':
  297. state->flags |= FMT_FORMAT_ALT_FORM;
  298. state->modifier = FMT_MODIFIER_PTR;
  299. __fallthrough;
  300. case 'x':
  301. state->flags |= FMT_FORMAT_LOWER;
  302. __fallthrough;
  303. case 'X':
  304. state->base = 16;
  305. state->specifier = FMT_SPECIFIER_INT;
  306. break;
  307. case 'c':
  308. state->specifier = FMT_SPECIFIER_CHAR;
  309. break;
  310. case 's':
  311. state->specifier = FMT_SPECIFIER_STR;
  312. break;
  313. case 'n':
  314. state->specifier = FMT_SPECIFIER_NRCHARS;
  315. break;
  316. case '%':
  317. state->specifier = FMT_SPECIFIER_PERCENT;
  318. break;
  319. default:
  320. state->specifier = FMT_SPECIFIER_INVALID;
  321. fmt_xprintf_state_restore_format(state);
  322. break;
  323. }
  324. }
  325. static int
  326. fmt_xprintf_state_consume(struct fmt_xprintf_state *state)
  327. {
  328. char c;
  329. c = fmt_consume(&state->format);
  330. if (c == '\0') {
  331. return ENOENT;
  332. }
  333. if (c != '%') {
  334. fmt_xprintf_state_produce_raw_char(state, c);
  335. return EAGAIN;
  336. }
  337. fmt_xprintf_state_consume_flags(state);
  338. fmt_xprintf_state_consume_width(state);
  339. fmt_xprintf_state_consume_precision(state);
  340. fmt_xprintf_state_consume_modifier(state);
  341. fmt_xprintf_state_consume_specifier(state);
  342. return 0;
  343. }
  344. static void
  345. fmt_xprintf_state_produce_int(struct fmt_xprintf_state *state)
  346. {
  347. char c, sign, tmp[FMT_MAX_NUM_SIZE];
  348. unsigned int r, mask, shift;
  349. unsigned long long n;
  350. int i;
  351. switch (state->modifier) {
  352. case FMT_MODIFIER_CHAR:
  353. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  354. n = (signed char)va_arg(state->ap, int);
  355. } else {
  356. n = (unsigned char)va_arg(state->ap, int);
  357. }
  358. break;
  359. case FMT_MODIFIER_SHORT:
  360. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  361. n = (short)va_arg(state->ap, int);
  362. } else {
  363. n = (unsigned short)va_arg(state->ap, int);
  364. }
  365. break;
  366. case FMT_MODIFIER_LONG:
  367. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  368. n = va_arg(state->ap, long);
  369. } else {
  370. n = va_arg(state->ap, unsigned long);
  371. }
  372. break;
  373. case FMT_MODIFIER_LONGLONG:
  374. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  375. n = va_arg(state->ap, long long);
  376. } else {
  377. n = va_arg(state->ap, unsigned long long);
  378. }
  379. break;
  380. case FMT_MODIFIER_PTR:
  381. n = (uintptr_t)va_arg(state->ap, void *);
  382. break;
  383. case FMT_MODIFIER_SIZE:
  384. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  385. n = va_arg(state->ap, ssize_t);
  386. } else {
  387. n = va_arg(state->ap, size_t);
  388. }
  389. break;
  390. case FMT_MODIFIER_PTRDIFF:
  391. n = va_arg(state->ap, ptrdiff_t);
  392. break;
  393. default:
  394. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  395. n = va_arg(state->ap, int);
  396. } else {
  397. n = va_arg(state->ap, unsigned int);
  398. }
  399. break;
  400. }
  401. if ((state->flags & FMT_FORMAT_LEFT_JUSTIFY) || (state->precision >= 0)) {
  402. state->flags &= ~FMT_FORMAT_ZERO_PAD;
  403. }
  404. sign = '\0';
  405. if (state->flags & FMT_FORMAT_ALT_FORM) {
  406. /* '0' for octal */
  407. state->width--;
  408. /* '0x' or '0X' for hexadecimal */
  409. if (state->base == 16) {
  410. state->width--;
  411. }
  412. } else if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  413. if ((long long)n < 0) {
  414. sign = '-';
  415. state->width--;
  416. n = -(long long)n;
  417. } else if (state->flags & FMT_FORMAT_SIGN) {
  418. /* FMT_FORMAT_SIGN must precede FMT_FORMAT_BLANK. */
  419. sign = '+';
  420. state->width--;
  421. } else if (state->flags & FMT_FORMAT_BLANK) {
  422. sign = ' ';
  423. state->width--;
  424. }
  425. }
  426. /* Conversion, in reverse order */
  427. i = 0;
  428. if (n == 0) {
  429. if (state->precision != 0) {
  430. tmp[i] = '0';
  431. i++;
  432. }
  433. } else if (state->base == 10) {
  434. /*
  435. * Try to avoid 64 bits operations if the processor doesn't
  436. * support them. Note that even when using modulus and
  437. * division operators close to each other, the compiler may
  438. * forge two functions calls to compute the quotient and the
  439. * remainder, whereas processor instructions are generally
  440. * correctly used once, giving both results at once, through
  441. * plain or reciprocal division.
  442. */
  443. #ifndef __LP64__
  444. if (state->modifier == FMT_MODIFIER_LONGLONG) {
  445. #endif /* __LP64__ */
  446. do {
  447. r = n % 10;
  448. n /= 10;
  449. tmp[i] = fmt_digits[r];
  450. i++;
  451. } while (n != 0);
  452. #ifndef __LP64__
  453. } else {
  454. unsigned long m;
  455. m = (unsigned long)n;
  456. do {
  457. r = m % 10;
  458. m /= 10;
  459. tmp[i] = fmt_digits[r];
  460. i++;
  461. } while (m != 0);
  462. }
  463. #endif /* __LP64__ */
  464. } else {
  465. mask = state->base - 1;
  466. shift = (state->base == 8) ? 3 : 4;
  467. do {
  468. r = n & mask;
  469. n >>= shift;
  470. tmp[i] = fmt_digits[r] | (state->flags & FMT_FORMAT_LOWER);
  471. i++;
  472. } while (n != 0);
  473. }
  474. if (i > state->precision) {
  475. state->precision = i;
  476. }
  477. state->width -= state->precision;
  478. if (!(state->flags & (FMT_FORMAT_LEFT_JUSTIFY | FMT_FORMAT_ZERO_PAD))) {
  479. while (state->width > 0) {
  480. state->width--;
  481. fmt_xprintf_state_produce_raw_char(state, ' ');
  482. }
  483. state->width--;
  484. }
  485. if (state->flags & FMT_FORMAT_ALT_FORM) {
  486. fmt_xprintf_state_produce_raw_char(state, '0');
  487. if (state->base == 16) {
  488. c = 'X' | (state->flags & FMT_FORMAT_LOWER);
  489. fmt_xprintf_state_produce_raw_char(state, c);
  490. }
  491. } else if (sign != '\0') {
  492. fmt_xprintf_state_produce_raw_char(state, sign);
  493. }
  494. if (!(state->flags & FMT_FORMAT_LEFT_JUSTIFY)) {
  495. c = (state->flags & FMT_FORMAT_ZERO_PAD) ? '0' : ' ';
  496. while (state->width > 0) {
  497. state->width--;
  498. fmt_xprintf_state_produce_raw_char(state, c);
  499. }
  500. state->width--;
  501. }
  502. while (i < state->precision) {
  503. state->precision--;
  504. fmt_xprintf_state_produce_raw_char(state, '0');
  505. }
  506. state->precision--;
  507. while (i > 0) {
  508. i--;
  509. fmt_xprintf_state_produce_raw_char(state, tmp[i]);
  510. }
  511. while (state->width > 0) {
  512. state->width--;
  513. fmt_xprintf_state_produce_raw_char(state, ' ');
  514. }
  515. state->width--;
  516. }
  517. static void
  518. fmt_xprintf_state_produce_char(struct fmt_xprintf_state *state)
  519. {
  520. char c;
  521. c = va_arg(state->ap, int);
  522. if (!(state->flags & FMT_FORMAT_LEFT_JUSTIFY)) {
  523. for (;;) {
  524. state->width--;
  525. if (state->width <= 0) {
  526. break;
  527. }
  528. fmt_xprintf_state_produce_raw_char(state, ' ');
  529. }
  530. }
  531. fmt_xprintf_state_produce_raw_char(state, c);
  532. for (;;) {
  533. state->width--;
  534. if (state->width <= 0) {
  535. break;
  536. }
  537. fmt_xprintf_state_produce_raw_char(state, ' ');
  538. }
  539. }
  540. static void
  541. fmt_xprintf_state_produce_str(struct fmt_xprintf_state *state)
  542. {
  543. int i, len;
  544. char *s;
  545. s = va_arg(state->ap, char *);
  546. if (s == NULL) {
  547. s = "(null)";
  548. }
  549. for (len = 0; s[len] != '\0'; len++) {
  550. if (len == state->precision) {
  551. break;
  552. }
  553. }
  554. if (!(state->flags & FMT_FORMAT_LEFT_JUSTIFY)) {
  555. while (len < state->width) {
  556. state->width--;
  557. fmt_xprintf_state_produce_raw_char(state, ' ');
  558. }
  559. }
  560. for (i = 0; i < len; i++) {
  561. fmt_xprintf_state_produce_raw_char(state, *s);
  562. s++;
  563. }
  564. while (len < state->width) {
  565. state->width--;
  566. fmt_xprintf_state_produce_raw_char(state, ' ');
  567. }
  568. }
  569. static void
  570. fmt_xprintf_state_produce_nrchars(struct fmt_xprintf_state *state)
  571. {
  572. if (state->modifier == FMT_MODIFIER_CHAR) {
  573. signed char *ptr = va_arg(state->ap, signed char *);
  574. *ptr = state->nr_chars;
  575. } else if (state->modifier == FMT_MODIFIER_SHORT) {
  576. short *ptr = va_arg(state->ap, short *);
  577. *ptr = state->nr_chars;
  578. } else if (state->modifier == FMT_MODIFIER_LONG) {
  579. long *ptr = va_arg(state->ap, long *);
  580. *ptr = state->nr_chars;
  581. } else if (state->modifier == FMT_MODIFIER_LONGLONG) {
  582. long long *ptr = va_arg(state->ap, long long *);
  583. *ptr = state->nr_chars;
  584. } else if (state->modifier == FMT_MODIFIER_SIZE) {
  585. ssize_t *ptr = va_arg(state->ap, ssize_t *);
  586. *ptr = state->nr_chars;
  587. } else if (state->modifier == FMT_MODIFIER_PTRDIFF) {
  588. ptrdiff_t *ptr = va_arg(state->ap, ptrdiff_t *);
  589. *ptr = state->nr_chars;
  590. } else {
  591. int *ptr = va_arg(state->ap, int *);
  592. *ptr = state->nr_chars;
  593. }
  594. }
  595. static void
  596. fmt_xprintf_state_produce(struct fmt_xprintf_state *state)
  597. {
  598. switch (state->specifier) {
  599. case FMT_SPECIFIER_INT:
  600. fmt_xprintf_state_produce_int(state);
  601. break;
  602. case FMT_SPECIFIER_CHAR:
  603. fmt_xprintf_state_produce_char(state);
  604. break;
  605. case FMT_SPECIFIER_STR:
  606. fmt_xprintf_state_produce_str(state);
  607. break;
  608. case FMT_SPECIFIER_NRCHARS:
  609. fmt_xprintf_state_produce_nrchars(state);
  610. break;
  611. case FMT_SPECIFIER_PERCENT:
  612. case FMT_SPECIFIER_INVALID:
  613. fmt_xprintf_state_produce_raw_char(state, '%');
  614. break;
  615. }
  616. }
  617. int
  618. fmt_vxprintf(struct fmt_write_op *op, const char *format, va_list ap)
  619. {
  620. struct fmt_xprintf_state state;
  621. int error;
  622. fmt_xprintf_state_init(&state, op, format, ap);
  623. for (;;) {
  624. error = fmt_xprintf_state_consume(&state);
  625. if (error == EAGAIN) {
  626. continue;
  627. } else if (error) {
  628. break;
  629. }
  630. fmt_xprintf_state_produce(&state);
  631. }
  632. return fmt_xprintf_state_finalize(&state);
  633. }
  634. int
  635. fmt_xprintf(struct fmt_write_op *op, const char *format, ...)
  636. {
  637. va_list ap;
  638. int ret;
  639. va_start(ap, format);
  640. ret = fmt_vxprintf(op, format, ap);
  641. va_end(ap);
  642. return ret;
  643. }
  644. static void
  645. sprintf_putc(void *data, int ch)
  646. {
  647. char **ptr;
  648. ptr = data;
  649. **ptr = ch;
  650. (*ptr)++;
  651. }
  652. int
  653. fmt_sprintf(char *str, const char *format, ...)
  654. {
  655. va_list ap;
  656. int length;
  657. va_start(ap, format);
  658. length = fmt_vsprintf(str, format, ap);
  659. va_end(ap);
  660. return length;
  661. }
  662. int
  663. fmt_vsprintf(char *str, const char *format, va_list ap)
  664. {
  665. struct fmt_write_op op;
  666. int ret;
  667. op.data = &str;
  668. op.putc = sprintf_putc;
  669. ret = fmt_vxprintf(&op, format, ap);
  670. *str = '\0';
  671. return ret;
  672. }
  673. int
  674. fmt_snprintf(char *str, size_t size, const char *format, ...)
  675. {
  676. va_list ap;
  677. int length;
  678. va_start(ap, format);
  679. length = fmt_vsnprintf(str, size, format, ap);
  680. va_end(ap);
  681. return length;
  682. }
  683. struct snprintf_data {
  684. char *cur;
  685. char *end;
  686. };
  687. static void
  688. snprintf_putc(void *ptr, int ch)
  689. {
  690. struct snprintf_data *data;
  691. data = ptr;
  692. if (data->cur < data->end) {
  693. *data->cur++ = ch;
  694. }
  695. }
  696. int
  697. fmt_vsnprintf(char *str, size_t size, const char *format, va_list ap)
  698. {
  699. struct fmt_write_op op;
  700. struct snprintf_data data;
  701. int ret;
  702. data.cur = str;
  703. data.end = str + size;
  704. op.data = &data;
  705. op.putc = snprintf_putc;
  706. ret = fmt_vxprintf(&op, format, ap);
  707. if (ret >= 0 && (size_t)ret < size) {
  708. *data.cur = '\0';
  709. }
  710. return ret;
  711. }
  712. static char
  713. fmt_atoi(char c)
  714. {
  715. assert(fmt_isxdigit(c));
  716. if (fmt_isdigit(c)) {
  717. return c - '0';
  718. } else if (c >= 'a' && c <= 'f') {
  719. return 10 + (c - 'a');
  720. } else {
  721. return 10 + (c - 'A');
  722. }
  723. }
  724. static bool
  725. fmt_isspace(char c)
  726. {
  727. if (c == ' ') {
  728. return true;
  729. }
  730. if ((c >= '\t') && (c <= '\f')) {
  731. return true;
  732. }
  733. return false;
  734. }
  735. static void
  736. fmt_xscanf_state_init(struct fmt_xscanf_state *state,
  737. struct fmt_read_op *op,
  738. const char *format, va_list ap)
  739. {
  740. state->format = format;
  741. state->flags = 0;
  742. state->width = 0;
  743. va_copy(state->ap, ap);
  744. state->nr_convs = 0;
  745. state->read_op = *op;
  746. state->nr_chars = 0;
  747. state->cur_ch = FMT_NOPUTBACK;
  748. }
  749. static int
  750. fmt_xscanf_state_finalize(struct fmt_xscanf_state *state)
  751. {
  752. va_end(state->ap);
  753. return state->nr_convs;
  754. }
  755. static void
  756. fmt_xscanf_state_report_conv(struct fmt_xscanf_state *state)
  757. {
  758. if (state->nr_convs == EOF) {
  759. state->nr_convs = 1;
  760. return;
  761. }
  762. state->nr_convs++;
  763. }
  764. static void
  765. fmt_xscanf_state_report_error(struct fmt_xscanf_state *state)
  766. {
  767. if (state->nr_convs != 0) {
  768. return;
  769. }
  770. state->nr_convs = EOF;
  771. }
  772. static void
  773. fmt_xscanf_state_putback(struct fmt_xscanf_state *state, int ch)
  774. {
  775. state->cur_ch = ch;
  776. }
  777. static int
  778. fmt_xscanf_state_consume_char(struct fmt_xscanf_state *state)
  779. {
  780. int ch;
  781. if (state->cur_ch != FMT_NOPUTBACK) {
  782. ch = state->cur_ch;
  783. state->cur_ch = FMT_NOPUTBACK;
  784. } else {
  785. ch = state->read_op.getc(state->read_op.data);
  786. }
  787. ++state->nr_chars;
  788. return ch;
  789. }
  790. static void
  791. fmt_xscanf_state_skip_space(struct fmt_xscanf_state *state)
  792. {
  793. int ch;
  794. for (;;) {
  795. ch = fmt_xscanf_state_consume_char(state);
  796. if (ch < 0 || !fmt_isspace(ch)) {
  797. fmt_xscanf_state_putback(state, ch);
  798. break;
  799. }
  800. }
  801. }
  802. static char
  803. fmt_xscanf_state_consume_string(struct fmt_xscanf_state *state)
  804. {
  805. char c;
  806. c = fmt_xscanf_state_consume_char(state);
  807. if (state->flags & FMT_FORMAT_CHECK_WIDTH) {
  808. if (state->width == 0) {
  809. c = EOF;
  810. } else {
  811. state->width--;
  812. }
  813. }
  814. return c;
  815. }
  816. static void
  817. fmt_xscanf_state_restore_string(struct fmt_xscanf_state *state, int ch)
  818. {
  819. assert(state->cur_ch == FMT_NOPUTBACK);
  820. state->cur_ch = ch;
  821. }
  822. static char
  823. fmt_xscanf_state_consume_format(struct fmt_xscanf_state *state)
  824. {
  825. return *state->format++;
  826. }
  827. static void
  828. fmt_xscanf_state_restore_format(struct fmt_xscanf_state *state)
  829. {
  830. --state->format;
  831. }
  832. static void
  833. fmt_xscanf_state_consume_flags(struct fmt_xscanf_state *state)
  834. {
  835. bool found;
  836. char c;
  837. found = true;
  838. state->flags = 0;
  839. do {
  840. c = fmt_xscanf_state_consume_format(state);
  841. switch (c) {
  842. case '*':
  843. state->flags |= FMT_FORMAT_DISCARD;
  844. break;
  845. default:
  846. found = false;
  847. break;
  848. }
  849. } while (found);
  850. fmt_xscanf_state_restore_format(state);
  851. }
  852. static void
  853. fmt_xscanf_state_consume_width(struct fmt_xscanf_state *state)
  854. {
  855. char c;
  856. state->width = 0;
  857. for (;;) {
  858. c = fmt_xscanf_state_consume_format(state);
  859. if (!fmt_isdigit(c)) {
  860. break;
  861. }
  862. state->width = state->width * 10 + (c - '0');
  863. }
  864. if (state->width != 0) {
  865. state->flags |= FMT_FORMAT_CHECK_WIDTH;
  866. }
  867. fmt_xscanf_state_restore_format(state);
  868. }
  869. static void
  870. fmt_xscanf_state_consume_modifier(struct fmt_xscanf_state *state)
  871. {
  872. char c, c2;
  873. c = fmt_xscanf_state_consume_format(state);
  874. switch (c) {
  875. case 'h':
  876. case 'l':
  877. c2 = fmt_xscanf_state_consume_format(state);
  878. if (c == c2) {
  879. state->modifier = (c == 'h') ? FMT_MODIFIER_CHAR
  880. : FMT_MODIFIER_LONGLONG;
  881. } else {
  882. state->modifier = (c == 'h') ? FMT_MODIFIER_SHORT
  883. : FMT_MODIFIER_LONG;
  884. fmt_xscanf_state_restore_format(state);
  885. }
  886. break;
  887. case 'z':
  888. state->modifier = FMT_MODIFIER_SIZE;
  889. break;
  890. case 't':
  891. state->modifier = FMT_MODIFIER_PTRDIFF;
  892. break;
  893. default:
  894. state->modifier = FMT_MODIFIER_NONE;
  895. fmt_xscanf_state_restore_format(state);
  896. break;
  897. }
  898. }
  899. static void
  900. fmt_xscanf_state_consume_specifier(struct fmt_xscanf_state *state)
  901. {
  902. char c;
  903. c = fmt_xscanf_state_consume_format(state);
  904. switch (c) {
  905. case 'i':
  906. state->base = 0;
  907. state->flags |= FMT_FORMAT_CONV_SIGNED;
  908. state->specifier = FMT_SPECIFIER_INT;
  909. break;
  910. case 'd':
  911. state->flags |= FMT_FORMAT_CONV_SIGNED;
  912. __fallthrough;
  913. case 'u':
  914. state->base = 10;
  915. state->specifier = FMT_SPECIFIER_INT;
  916. break;
  917. case 'o':
  918. state->base = 8;
  919. state->specifier = FMT_SPECIFIER_INT;
  920. break;
  921. case 'p':
  922. state->modifier = FMT_MODIFIER_PTR;
  923. __fallthrough;
  924. case 'x':
  925. case 'X':
  926. state->base = 16;
  927. state->specifier = FMT_SPECIFIER_INT;
  928. break;
  929. case 'c':
  930. state->specifier = FMT_SPECIFIER_CHAR;
  931. break;
  932. case 's':
  933. state->specifier = FMT_SPECIFIER_STR;
  934. break;
  935. case 'n':
  936. state->specifier = FMT_SPECIFIER_NRCHARS;
  937. break;
  938. case '%':
  939. state->specifier = FMT_SPECIFIER_PERCENT;
  940. break;
  941. default:
  942. state->specifier = FMT_SPECIFIER_INVALID;
  943. fmt_xscanf_state_restore_format(state);
  944. break;
  945. }
  946. }
  947. static int
  948. fmt_xscanf_state_discard_char(struct fmt_xscanf_state *state, char c)
  949. {
  950. char c2;
  951. if (fmt_isspace(c)) {
  952. fmt_xscanf_state_skip_space(state);
  953. return 0;
  954. }
  955. c2 = fmt_xscanf_state_consume_string(state);
  956. if (c != c2) {
  957. if ((c2 == '\0') && (state->nr_convs == 0)) {
  958. state->nr_convs = EOF;
  959. }
  960. return EINVAL;
  961. }
  962. return 0;
  963. }
  964. static int
  965. fmt_xscanf_state_consume(struct fmt_xscanf_state *state)
  966. {
  967. int error;
  968. char c;
  969. state->flags = 0;
  970. c = fmt_xscanf_state_consume_format(state);
  971. if (c == '\0') {
  972. return ENOENT;
  973. }
  974. if (c != '%') {
  975. error = fmt_xscanf_state_discard_char(state, c);
  976. if (error) {
  977. return error;
  978. }
  979. return EAGAIN;
  980. }
  981. fmt_xscanf_state_consume_flags(state);
  982. fmt_xscanf_state_consume_width(state);
  983. fmt_xscanf_state_consume_modifier(state);
  984. fmt_xscanf_state_consume_specifier(state);
  985. return 0;
  986. }
  987. static int
  988. fmt_xscanf_state_produce_int(struct fmt_xscanf_state *state)
  989. {
  990. unsigned long long n, m, tmp;
  991. char c, buf[FMT_MAX_NUM_SIZE];
  992. bool negative;
  993. size_t i;
  994. negative = 0;
  995. fmt_xscanf_state_skip_space(state);
  996. c = fmt_xscanf_state_consume_string(state);
  997. if (c == '-') {
  998. negative = true;
  999. c = fmt_xscanf_state_consume_string(state);
  1000. }
  1001. if (c == '0') {
  1002. c = fmt_xscanf_state_consume_string(state);
  1003. if ((c == 'x') || (c == 'X')) {
  1004. if (state->base == 0) {
  1005. state->base = 16;
  1006. }
  1007. if (state->base == 16) {
  1008. c = fmt_xscanf_state_consume_string(state);
  1009. } else {
  1010. fmt_xscanf_state_restore_string(state, c);
  1011. c = '0';
  1012. }
  1013. } else {
  1014. if (state->base == 0) {
  1015. state->base = 8;
  1016. }
  1017. if (state->base != 8) {
  1018. fmt_xscanf_state_restore_string(state, c);
  1019. c = '0';
  1020. }
  1021. }
  1022. }
  1023. i = 0;
  1024. while (c != '\0') {
  1025. if (state->base == 8) {
  1026. if (!((c >= '0') && (c <= '7'))) {
  1027. break;
  1028. }
  1029. } else if (state->base == 16) {
  1030. if (!fmt_isxdigit(c)) {
  1031. break;
  1032. }
  1033. } else {
  1034. if (!fmt_isdigit(c)) {
  1035. break;
  1036. }
  1037. }
  1038. /* XXX Standard sscanf provides no way to cleanly handle overflows */
  1039. if (i < (ARRAY_SIZE(buf) - 1)) {
  1040. buf[i] = c;
  1041. } else if (i == (ARRAY_SIZE(buf) - 1)) {
  1042. strcpy(buf, "1");
  1043. negative = true;
  1044. }
  1045. i++;
  1046. c = fmt_xscanf_state_consume_string(state);
  1047. }
  1048. fmt_xscanf_state_restore_string(state, c);
  1049. if (state->flags & FMT_FORMAT_DISCARD) {
  1050. return 0;
  1051. }
  1052. if (i == 0) {
  1053. if (c == '\0') {
  1054. fmt_xscanf_state_report_error(state);
  1055. return EINVAL;
  1056. }
  1057. buf[0] = '0';
  1058. i = 1;
  1059. }
  1060. if (i < ARRAY_SIZE(buf)) {
  1061. buf[i] = '\0';
  1062. i--;
  1063. } else {
  1064. i = strlen(buf) - 1;
  1065. }
  1066. n = 0;
  1067. #ifndef __LP64__
  1068. if (state->modifier == FMT_MODIFIER_LONGLONG) {
  1069. #endif /* __LP64__ */
  1070. m = 1;
  1071. tmp = 0;
  1072. while (&buf[i] >= buf) {
  1073. tmp += fmt_atoi(buf[i]) * m;
  1074. if (tmp < n) {
  1075. n = 1;
  1076. negative = true;
  1077. break;
  1078. }
  1079. n = tmp;
  1080. m *= state->base;
  1081. i--;
  1082. }
  1083. #ifndef __LP64__
  1084. } else {
  1085. unsigned long _n, _m, _tmp;
  1086. _n = 0;
  1087. _m = 1;
  1088. _tmp = 0;
  1089. while (&buf[i] >= buf) {
  1090. _tmp += fmt_atoi(buf[i]) * _m;
  1091. if (_tmp < _n) {
  1092. _n = 1;
  1093. negative = true;
  1094. break;
  1095. }
  1096. _n = _tmp;
  1097. _m *= state->base;
  1098. i--;
  1099. }
  1100. n = _n;
  1101. }
  1102. #endif /* __LP64__ */
  1103. if (negative) {
  1104. n = -n;
  1105. }
  1106. switch (state->modifier) {
  1107. case FMT_MODIFIER_CHAR:
  1108. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  1109. *va_arg(state->ap, char *) = n;
  1110. } else {
  1111. *va_arg(state->ap, unsigned char *) = n;
  1112. }
  1113. break;
  1114. case FMT_MODIFIER_SHORT:
  1115. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  1116. *va_arg(state->ap, short *) = n;
  1117. } else {
  1118. *va_arg(state->ap, unsigned short *) = n;
  1119. }
  1120. break;
  1121. case FMT_MODIFIER_LONG:
  1122. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  1123. *va_arg(state->ap, long *) = n;
  1124. } else {
  1125. *va_arg(state->ap, unsigned long *) = n;
  1126. }
  1127. break;
  1128. case FMT_MODIFIER_LONGLONG:
  1129. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  1130. *va_arg(state->ap, long long *) = n;
  1131. } else {
  1132. *va_arg(state->ap, unsigned long long *) = n;
  1133. }
  1134. break;
  1135. case FMT_MODIFIER_PTR:
  1136. *va_arg(state->ap, uintptr_t *) = n;
  1137. break;
  1138. case FMT_MODIFIER_SIZE:
  1139. *va_arg(state->ap, size_t *) = n;
  1140. break;
  1141. case FMT_MODIFIER_PTRDIFF:
  1142. *va_arg(state->ap, ptrdiff_t *) = n;
  1143. break;
  1144. default:
  1145. if (state->flags & FMT_FORMAT_CONV_SIGNED) {
  1146. *va_arg(state->ap, int *) = n;
  1147. } else {
  1148. *va_arg(state->ap, unsigned int *) = n;
  1149. }
  1150. }
  1151. fmt_xscanf_state_report_conv(state);
  1152. return 0;
  1153. }
  1154. static int
  1155. fmt_xscanf_state_produce_char(struct fmt_xscanf_state *state)
  1156. {
  1157. char c, *dest;
  1158. int i, width;
  1159. if (state->flags & FMT_FORMAT_DISCARD) {
  1160. dest = NULL;
  1161. } else {
  1162. dest = va_arg(state->ap, char *);
  1163. }
  1164. if (state->flags & FMT_FORMAT_CHECK_WIDTH) {
  1165. width = state->width;
  1166. } else {
  1167. width = 1;
  1168. }
  1169. for (i = 0; i < width; i++) {
  1170. c = fmt_xscanf_state_consume_string(state);
  1171. if ((c == '\0') || (c == EOF)) {
  1172. break;
  1173. }
  1174. if (dest != NULL) {
  1175. *dest = c;
  1176. dest++;
  1177. }
  1178. }
  1179. if (i < width) {
  1180. fmt_xscanf_state_restore_string(state, c);
  1181. }
  1182. if ((dest != NULL) && (i != 0)) {
  1183. fmt_xscanf_state_report_conv(state);
  1184. }
  1185. return 0;
  1186. }
  1187. static int
  1188. fmt_xscanf_state_produce_str(struct fmt_xscanf_state *state)
  1189. {
  1190. int orig, off;
  1191. char c, dummy, *dest;
  1192. orig = state->nr_chars;
  1193. fmt_xscanf_state_skip_space(state);
  1194. if (state->flags & FMT_FORMAT_DISCARD) {
  1195. dest = &dummy;
  1196. off = 0;
  1197. } else {
  1198. dest = va_arg(state->ap, char *);
  1199. off = 1;
  1200. }
  1201. for (;;) {
  1202. c = fmt_xscanf_state_consume_string(state);
  1203. if ((c == '\0') || (c == ' ') || (c == EOF)) {
  1204. break;
  1205. }
  1206. *dest = c;
  1207. dest += off;
  1208. }
  1209. fmt_xscanf_state_restore_string(state, c);
  1210. if (state->nr_chars == orig) {
  1211. fmt_xscanf_state_report_error(state);
  1212. return EINVAL;
  1213. }
  1214. if (dest != &dummy) {
  1215. *dest = '\0';
  1216. fmt_xscanf_state_report_conv(state);
  1217. }
  1218. return 0;
  1219. }
  1220. static int
  1221. fmt_xscanf_state_produce_nrchars(struct fmt_xscanf_state *state)
  1222. {
  1223. *va_arg(state->ap, int *) = state->nr_chars;
  1224. return 0;
  1225. }
  1226. static int
  1227. fmt_xscanf_state_produce(struct fmt_xscanf_state *state)
  1228. {
  1229. switch (state->specifier) {
  1230. case FMT_SPECIFIER_INT:
  1231. return fmt_xscanf_state_produce_int(state);
  1232. case FMT_SPECIFIER_CHAR:
  1233. return fmt_xscanf_state_produce_char(state);
  1234. case FMT_SPECIFIER_STR:
  1235. return fmt_xscanf_state_produce_str(state);
  1236. case FMT_SPECIFIER_NRCHARS:
  1237. return fmt_xscanf_state_produce_nrchars(state);
  1238. case FMT_SPECIFIER_PERCENT:
  1239. fmt_xscanf_state_skip_space(state);
  1240. return fmt_xscanf_state_discard_char(state, '%');
  1241. default:
  1242. fmt_xscanf_state_report_error(state);
  1243. return EINVAL;
  1244. }
  1245. }
  1246. int
  1247. fmt_vxscanf(struct fmt_read_op *op, const char *format, va_list ap)
  1248. {
  1249. struct fmt_xscanf_state state;
  1250. int error;
  1251. fmt_xscanf_state_init(&state, op, format, ap);
  1252. for (;;) {
  1253. error = fmt_xscanf_state_consume(&state);
  1254. if (error == EAGAIN) {
  1255. continue;
  1256. } else if (error) {
  1257. break;
  1258. }
  1259. error = fmt_xscanf_state_produce(&state);
  1260. if (error) {
  1261. break;
  1262. }
  1263. }
  1264. return fmt_xscanf_state_finalize(&state);
  1265. }
  1266. int
  1267. fmt_xscanf(struct fmt_read_op *op, const char *format, ...)
  1268. {
  1269. va_list ap;
  1270. int ret;
  1271. va_start(ap, format);
  1272. ret = fmt_vxscanf(op, format, ap);
  1273. va_end(ap);
  1274. return ret;
  1275. }
  1276. static int
  1277. sscanf_getc(void *ptr)
  1278. {
  1279. char **cur;
  1280. int ch;
  1281. cur = ptr;
  1282. ch = **cur;
  1283. if (ch != '\0') {
  1284. (*cur)++;
  1285. }
  1286. return ch;
  1287. }
  1288. int
  1289. fmt_sscanf(const char *str, const char *format, ...)
  1290. {
  1291. va_list ap;
  1292. int ret;
  1293. va_start(ap, format);
  1294. ret = fmt_vsscanf(str, format, ap);
  1295. va_end(ap);
  1296. return ret;
  1297. }
  1298. int
  1299. fmt_vsscanf(const char *str, const char *format, va_list ap)
  1300. {
  1301. struct fmt_read_op op;
  1302. op.getc = sscanf_getc;
  1303. op.data = (void *)&str;
  1304. return fmt_vxscanf(&op, format, ap);
  1305. }