diff options
author | Michael Forney <mforney@mforney.org> | 2020-03-18 15:04:14 -0700 |
---|---|---|
committer | Michael Forney <mforney@mforney.org> | 2020-03-18 15:50:33 -0700 |
commit | 84da43b2235b1ec82b84ef8100c05097bc3ca4f5 (patch) | |
tree | fcc7177f44e6aa14b0e61462fd45c217d3eefa60 | |
parent | 998d77adf854fd3f3ce630372068fd3aa1edb2c2 (diff) | |
download | cproc-84da43b2235b1ec82b84ef8100c05097bc3ca4f5.tar.xz |
pp: Fix possible use-after-free
Restructure the loop slightly to avoid accessing the macro identifier
token after we have checked for the open parenthesis; it might
belong to a previous macro argument that has since been freed.
-rw-r--r-- | pp.c | 14 |
1 files changed, 9 insertions, 5 deletions
@@ -428,16 +428,15 @@ expand(struct token *t) arg = xreallocarray(NULL, m->nparam, sizeof(*arg)); } else { arg = NULL; - t = rawnext(); } - for (i = 0; i < m->nparam && t->kind != TRPAREN; ++i) { + t = rawnext(); + for (i = 0; i < m->nparam; ++i) { if (m->param[i].flags & PARAMSTR) { str = (struct array){0}; arrayaddbuf(&str, "\"", 1); } arg[i].ntoken = 0; for (;;) { - t = rawnext(); if (t->kind == TEOF) error(&t->loc, "EOF when reading macro parameters"); if (macrodepth <= depth) { @@ -456,6 +455,7 @@ expand(struct token *t) arrayaddbuf(&tok, t, sizeof(*t)); ++arg[i].ntoken; } + t = rawnext(); } if (m->param[i].flags & PARAMSTR) { arrayaddbuf(&str, "\"", 2); @@ -464,10 +464,14 @@ expand(struct token *t) .lit = str.val, }; } + if (t->kind == TRPAREN) + break; + t = rawnext(); } - if (i < m->nparam) + if (i + 1 < m->nparam) error(&t->loc, "not enough arguments for macro '%s'", m->name); - tokencheck(t, TRPAREN, "after macro arguments"); + if (t->kind != TRPAREN) + error(&t->loc, "too many arguments for macro '%s'", m->name); for (i = 0, t = tok.val; i < m->nparam; ++i) { if (m->param[i].flags & PARAMTOK) { arg[i].token = t; |