static void check_text(struct mdoc *m, int ln, int pos, char *p) { int c; size_t sz; for ( ; *p; p++, pos++) { sz = strcspn(p, "\t\\"); p += (int)sz; if ('\0' == *p) break; pos += (int)sz; if ('\t' == *p) { if ( ! (MDOC_LITERAL & m->flags)) mdoc_pmsg(m, ln, pos, MANDOCERR_BADTAB); continue; } if (0 == (c = mandoc_special(p))) { mdoc_pmsg(m, ln, pos, MANDOCERR_BADESCAPE); continue; } p += c - 1; pos += c - 1; } }
int mandoc_special(char *p) { int len, i; char term; char *sv; len = 0; term = '\0'; sv = p; assert('\\' == *p); p++; switch (*p++) { #if 0 case ('Z'): /* FALLTHROUGH */ case ('X'): /* FALLTHROUGH */ case ('x'): /* FALLTHROUGH */ case ('S'): /* FALLTHROUGH */ case ('R'): /* FALLTHROUGH */ case ('N'): /* FALLTHROUGH */ case ('l'): /* FALLTHROUGH */ case ('L'): /* FALLTHROUGH */ case ('H'): /* FALLTHROUGH */ case ('h'): /* FALLTHROUGH */ case ('D'): /* FALLTHROUGH */ case ('C'): /* FALLTHROUGH */ case ('b'): /* FALLTHROUGH */ case ('B'): /* FALLTHROUGH */ case ('a'): /* FALLTHROUGH */ case ('A'): if (*p++ != '\'') return(0); term = '\''; break; #endif case ('h'): /* FALLTHROUGH */ case ('v'): /* FALLTHROUGH */ case ('s'): if (ASCII_HYPH == *p) *p = '-'; i = 0; if ('+' == *p || '-' == *p) { p++; i = 1; } switch (*p++) { case ('('): len = 2; break; case ('['): term = ']'; break; case ('\''): term = '\''; break; case ('0'): i = 1; /* FALLTHROUGH */ default: len = 1; p--; break; } if (ASCII_HYPH == *p) *p = '-'; if ('+' == *p || '-' == *p) { if (i) return(0); p++; } /* Handle embedded numerical subexp or escape. */ if ('(' == *p) { while (*p && ')' != *p) if ('\\' == *p++) { i = mandoc_special(--p); if (0 == i) return(0); p += i; } if (')' == *p++) break; return(0); } else if ('\\' == *p) { if (0 == (i = mandoc_special(p))) return(0); p += i; } break; #if 0 case ('Y'): /* FALLTHROUGH */ case ('V'): /* FALLTHROUGH */ case ('$'): /* FALLTHROUGH */ case ('n'): /* FALLTHROUGH */ #endif case ('k'): /* FALLTHROUGH */ case ('M'): /* FALLTHROUGH */ case ('m'): /* FALLTHROUGH */ case ('f'): /* FALLTHROUGH */ case ('F'): /* FALLTHROUGH */ case ('*'): switch (*p++) { case ('('): len = 2; break; case ('['): term = ']'; break; default: len = 1; p--; break; } break; case ('('): len = 2; break; case ('['): term = ']'; break; case ('z'): len = 1; if ('\\' == *p) { if (0 == (i = mandoc_special(p))) return(0); p += i; return(*p ? (int)(p - sv) : 0); } break; case ('o'): /* FALLTHROUGH */ case ('w'): if ('\'' == *p++) { term = '\''; break; } /* FALLTHROUGH */ default: len = 1; p--; break; } if (term) { for ( ; *p && term != *p; p++) if (ASCII_HYPH == *p) *p = '-'; return(*p ? (int)(p - sv) : 0); } for (i = 0; *p && i < len; i++, p++) if (ASCII_HYPH == *p) *p = '-'; return(i == len ? (int)(p - sv) : 0); }