/* $NetBSD: tgoto.c,v 1.21 2002/07/04 18:47:28 christos Exp $ */ /* * Copyright (c) 1980, 1993 * The Regents of the University of California. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by the University of * California, Berkeley and its contributors. * 4. Neither the name of the University nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include #ifndef lint #if 0 static char sccsid[] = "@(#)tgoto.c 8.1 (Berkeley) 6/4/93"; #else __RCSID("$NetBSD: tgoto.c,v 1.21 2002/07/04 18:47:28 christos Exp $"); #endif #endif /* not lint */ #include #include #include #include #include #define CTRL(c) ((c) & 037) #define MAXRETURNSIZE 64 char *UP; char *BC; /* * Routine to perform cursor addressing. * CM is a string containing printf type escapes to allow * cursor addressing. We start out ready to print the destination * line, and switch each time we print row or column. * The following escapes are defined for substituting row/column: * * %d as in printf * %2 like %2d * %3 like %3d * %. gives %c hacking special case characters * %+x like %c but adding x first * * The codes below affect the state but don't use up a value. * * %>xy if value > x add y * %r reverses row/column * %i increments row/column (for one origin indexing) * %% gives % * %B BCD (2 decimal digits encoded in one byte) * %D Delta Data (backwards bcd) * * all other characters are ``self-inserting''. */ char * tgoto(CM, destcol, destline) const char *CM; int destcol, destline; { static char result[MAXRETURNSIZE]; if (t_goto(NULL, CM, destcol, destline, result, MAXRETURNSIZE) >= 0) return result; else return ("OOPS"); } /* * New interface. Functionally the same as tgoto but uses the tinfo struct * to set UP and BC. The arg buffer is filled with the result string, limit * defines the maximum number of chars allowed in buffer. The function * returns 0 on success, -1 otherwise. */ int t_goto(info, CM, destcol, destline, buffer, limit) struct tinfo *info; const char *CM; int destcol; int destline; char *buffer; size_t limit; { char added[10]; const char *cp = CM; char *dp = buffer; int c; int oncol = 0; int which = destline; char *buf_lim = buffer + limit; char dig_buf[3 * sizeof(which)]; char *ap = added; char *eap = &added[sizeof(added) / sizeof(added[0])]; int k; /* CM is checked below */ _DIAGASSERT(buffer != NULL); if (info != NULL) { if (!UP) UP = info->up; if (!BC) BC = info->bc; } if (cp == 0) { errno = EINVAL; return -1; } added[0] = '\0'; while ((c = *cp++) != '\0') { if (c != '%' || ((c = *cp++) == '%')) { *dp++ = c; if (dp >= buf_lim) { errno = E2BIG; return -1; } continue; } switch (c) { #ifdef CM_N case 'n': destcol ^= 0140; destline ^= 0140; /* flip oncol here so it doesn't actually change */ oncol = 1 - oncol; break; #endif case '3': case '2': case 'd': /* Generate digits into temp buffer in reverse order */ k = 0; do dig_buf[k++] = which % 10 | '0'; while ((which /= 10) != 0); if (c != 'd') { c -= '0'; if (k > c) { errno = E2BIG; return -1; } while (k < c) dig_buf[k++] = '0'; } if (dp + k >= buf_lim) { errno = E2BIG; return -1; } /* then unwind into callers buffer */ do *dp++ = dig_buf[--k]; while (k); break; #ifdef CM_GT case '>': if (which > *cp++) which += *cp++; else cp++; continue; #endif case '+': which += *cp++; /* FALLTHROUGH */ case '.': /* * This code is worth scratching your head at for a * while. The idea is that various weird things can * happen to nulls, EOT's, tabs, and newlines by the * tty driver, arpanet, and so on, so we don't send * them if we can help it. * * Tab is taken out to get Ann Arbors to work, otherwise * when they go to column 9 we increment which is wrong * because bcd isn't continuous. We should take out * the rest too, or run the thing through more than * once until it doesn't make any of these, but that * would make termlib (and hence pdp-11 ex) bigger, * and also somewhat slower. This requires all * programs which use termlib to stty tabs so they * don't get expanded. They should do this anyway * because some terminals use ^I for other things, * like nondestructive space. */ if (which == 0 || which == CTRL('d') || /* which == '\t' || */ which == '\n') { if (oncol || UP) { /* Assumption: backspace works */ char *add = oncol ? (BC ? BC : "\b") : UP; /* * Loop needed because newline happens * to be the successor of tab. */ do { char *as = add; while ((*ap++ = *as++) != '\0') if (ap >= eap) { errno = E2BIG; return -1; } which++; } while (which == '\n'); } } *dp++ = which; if (dp >= buf_lim) { errno = E2BIG; return -1; } break; case 'r': oncol = 0; break; case 'i': destcol++; destline++; which++; continue; #ifdef CM_B case 'B': which = (which / 10 << 4) + which % 10; continue; #endif #ifdef CM_D case 'D': which = which - 2 * (which % 16); continue; #endif default: errno = EINVAL; return -1; } /* flip to other number... */ oncol = 1 - oncol; which = oncol ? destcol : destline; } if (dp + (ap - added) >= buf_lim) { errno = E2BIG; return -1; } *ap = '\0'; while ((*dp++ = *ap++) != '\0') continue; return 0; }