/* CONFIG.C -- Configuration Program for TVI 950
written in BDS C by Richard Conn
Program Name: CONFIG.C
Version: 1.2
Date: 6 Oct 82
Notes:
The purpose of this program is to configure the TVI 950 CRT
to respond in a manner desired by the user. Several of the
attributes of the TVI 950 are examined and set by this pgm.
These attributes include:
. Function Key Definitions
. User Line Display
. Scrolling (Smooth or Hard)
. Tab Stops
. Video Display (Black on White or vice-versa)
. Key Click (On or Off)
. Cursor Type
CONFIG runs in two basic modes -- Setup and Configure.
Setup mode allows the user to set the various parameters of the
terminal interactively. Configure programs the terminal. CONFIG
is invoked by the following forms:
CONFIG <-- Programs the terminal from its internal
configuration setting
CONFIG filename <-- Programs the terminal from the settings
given in the data file; if file
type is not specified, a type of
CFG is assumed; if the file is not
found on the current drive, drive A:
is examined (user areas not changed)
CONFIG /S <-- Enter Setup mode; initial setting values
are those already programmed into
CONFIG
CONFIG fname /S <-- Enter Setup mode with settings from the
indicated file
The data files created and read by this program and read by the
accompanying TINIT program conform to the following structure:
Line 1:
Variable Values Meaning
-------- ------ -------
cursor '0' to '4' Cursor Attributes
fkeys 'Y' or 'N' Program Function Keys
click '>' or '<' Key Click On or Off
scroll '8' or '9' Enable Smooth or Hard Scrolling
tabs 'Y' or 'N' Set Tab Stops
uline 'Y' or 'N' Set and Display User Line
video 'b' or 'd' Reverse or Normal Video
These characters are followed by <CR> <LF>
Line n:
If uline=YES then text of ulbuffer (User Line); length is ullen;
text is followed by <CR> <LF>
If tabs=YES then text of tabbuffer (contains <SP> and '.'); length
is tablen; text is followed by <CR> <LF>
If fkeys=YES then text of fkeybuffer (22 strings of up to 20 chars
each); length is fkeylen; text is followed by <CR> <LF>
/* Function Keys */
#define fctkey '|' /* Begin function key definition */
#define fctend 0x19 /* Ctrl-Y for end function key definition */
#define ctrlp 0x10 /* Control-P for char ESCAPE in Fct Key cmd */
/* User Line */
#define loadusr 'f' /* Load User Line */
#define dispusr 'g' /* Display User Line */
#define dispst 'h' /* Display Status Line */
main(argc,argv)
int argc;
char **argv;
{
char *argstr;
switch (argc) {
case 1 : program(); break;
case 2 : argstr = argv[1]; /* Pt to first arg */
if (*argstr == '/') { /* SETUP or HELP */
if (*++argstr == 'S') setup("");
else help(); /* HELP if /non-S */
}
else {
readfile(argstr);
program();
}
break;
case 3 : argstr = argv[2]; /* Pt to 2nd arg */
if ((*argstr == '/') && (*++argstr == 'S')) {
readfile(argv[1]);
setup(argv[1]);
}
else help();
break;
default : help(); break;
}
exit(TRUE);
} /* End of Mainline */
help() /* HELP File */
{
puts("\nCONFIG -- Configuration for TVI 950 Terminal");
puts("\n CONFIG is invoked by one of the following forms --");
puts("\n\tCONFIG\t\t<-- Configured by internal settings");
puts("\n\tCONFIG /S\t\t<-- User Sets Up Configuration");
puts("\n\tCONFIG filename.typ\t<-- Configured by file");
puts("\n\tCONFIG filename.typ /S\t<-- Set Up from file");
}
/********************************************************************
SETUP Routine to initialize a Configuration File
********************************************************************/
setup(filename) /* This is the master menu and command routine for setting up
the TVI 950 characteristics */
char *filename;
{
int exit;
char ch;
exit = FALSE; /* Set no exit for now */
cmnd(hscroll); /* Hard Scrolling for Displays */
cmnd(dispst); /* Display Status Line */
newscreen (filename); /* Display Banner and Menu */
/* Main Command Processing Loop */
do {
at (14,17," "); at (14,17,""); ch = response(echo);
switch (ch) {
case CR : newscreen(filename); break;
case 'C' : cursor=setcursor(); break;
case 'D' : status(); newscreen(filename); break;
case 'F' : fkeys=setfkeys(); newscreen(filename); break;
case 'K' : click=setclick(); break;
case 'P' : program(); newscreen(filename); break;
case 'S' : scroll=setscroll(); break;
case 'T' : tabs=settabs(); break;
case 'U' : uline=setuline(); break;
case 'V' : video=setvideo(); break;
case 'X' : exit = TRUE; break;
case 'Z' : readwrite(filename); break;
default : cmderror(ch); break;
}
} while (!exit);
cls(); /* Write exiting banner */
at (1,1,"CONFIG Exiting to CP/M");
}
newscreen(filename) /* Print menu of commands on screen */
char *filename;
{
int i;
cls(); /* Clear Screen */
cmnd('$'); /* Graphics ON */
at (2,21,"bkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkc"); /* Title Box */
at (3,21,"j"); at (3,61,"j");
at (4,21,"akkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkd");
at (6,3,"bkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkc"); /* Left Box */
at (13,3,"akkkkkkkkkkkkkkkkkkkkkkkkkkkkkkd");
at (6,46,"bkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkc"); /* Right Box */
at (13,46,"akkkkkkkkkkkkkkkkkkkkkkkkkkkkkkd");
for (i=7;i<13;i++) {
at (i,3,"j"); at (i,34,"j"); /* Left Box */
at (i,46,"j"); at (i,77,"j"); /* Right Box */
}
for (i=3;i<79;i++) at (17,i,"k"); /* Dividing Line */
cmnd('%'); /* Graphics OFF */
at (3,26,"");
printf("CONFIG for TVI 950, Version %d.%d",
version/10,version%10); /* Title Box */
at ( 7,5,"Commands to Set Attributes"); /* Left Box */
at ( 8,5,"Cmd Attribute Cmd Attribute");
at ( 9,5," C Cursor Type S Scrolling");
at (10,5," D Disp Status T Tabs");
at (11,5," F Fct Keys U User Line");
at (12,5," K Key Click V Video");
at ( 7,48,"Commands to Execute Functions"); /* Right Box */
at ( 8,48," Cmd Function");
at ( 9,48," P Program TVI 950");
at (10,48," X Exit to CP/M");
at (11,48," Z Read/Write File");
at (12,48,"<cr> Refresh Screen");
at (14,7,"Command?"); /* Command Line Prompt */
}
setcursor() /* Set cursor type */
{
char ch;
do {
clreos(); /* Clear Display Area */
at (18, 1,"Select Type of Cursor --");
at (19,10,"0 - No Cursor");
at (20,10,"1 - Blinking Block 2 - Steady Block");
at (21,10,"3 - Blinking Underline 4 - Steady Underline");
at (23,15,"Selection? "); ch=response(echo);
} while (('0' > ch) | (ch > '4'));
return(ch);
}
status() /* Display Status of CONFIG */
{
char ch;
int i;
cls();
at (1,33,"CONFIG Status Display");
cmnd('$'); /* Graphics ON */
at (1,32,"J"); at (1,54,"J");
at (2,32,"AKKKKKKKKKKKKKKKKKKKKKD");
cmnd('%'); /* Graphics OFF */
if (fkeys == YES) fkeystatus(); /* Print Function Key Information */
at (3,1,"User Line --");
if (uline == YES) printf("\n%s",ulbuffer);
else puts("\nUser Line is NOT DISPLAYED");
at (5,1,"Tab Settings are --");
if (tabs == YES) {
printf("\n");
for (i=0; i<tablen; i++) putchar(tabbuffer[i]);
at (7,1,"");
for (i=1; i<=tablen; i++) if (i%5 == 0) {
if (i == 5) printf(" 5");
else printf(" %d",i);
}
puts("\n");
}
else puts("\nTabs are NOT SET");
at (11,1,"Key Click is ");
if (click == clkon) puts("ON"); else puts("OFF");
at (12,1,"Cursor is ");
switch (cursor) {
case '0' : puts("NOT DISPLAYED"); break;
case '1' : puts("Blinking Block"); break;
case '2' : puts("Steady Block"); break;
case '3' : puts("Blinking Uline"); break;
default : puts("Steady Uline"); break;
}
at (13,1,"Scrolling is ");
if (scroll == sscroll) puts("Soft"); else puts("Hard");
at (14,1,"Video is ");
if (video == vidnorm) puts("Normal"); else puts("Reverse");
at (24,1,"Type Any Char to Cont -- "); ch=response(noecho);
}
setfkeys() /* This routine allows the user to define his function keys */
{
char ch, *lptr, fstr[21], *tstr;
int fnum, *numptr, start, i;
clreos(); /* Clear Display Area */
at (18,1,"Do you Wish to Initialize the Function Keys? ");
ch=response(noecho);
if (ch == YES) for (fnum=1;fnum<=22;fnum++) { /* Init Keys */
lptr = fkeybuffer + (fnum-1)*20;
*lptr++ = '1'; /* Transmit to computer */
*lptr = '\0'; /* Init to empty */
}
cls(); /* Clear Display Area */
at (1,34,"Function Key Programming");
cmnd('$'); /* Graphics ON */
at (1,33,"J"); at (1,58,"J");
at (2,33,"AKKKKKKKKKKKKKKKKKKKKKKKKD");
cmnd('%'); /* Graphics OFF */
fkeystatus(); /* Print current settings of keys */
/* Main Function Key Programming Command Loop */
do {
at (3,1,"Number of Function Key (0=Stop, 1-11=Fct)? ");
dots(2);
*numptr = 0; scanf("%d",numptr); fnum = *numptr;
if ((fnum < 1) | (fnum > 11)) fnum=0;
if (fnum != 0) {
at (4,1,"Is this Key Shifted (Y/N)? ");
ch=response(echo);
if (ch == YES) fnum += 11;
at (5,1,"String for this Key? ");
dots(19); inline(fstr,19);
lptr = fkeybuffer + (fnum-1)*20; /* Pt to key entry */
at (6,1,"Is this command locally executed? ");
ch=response(echo);
if (ch == YES) *lptr++ = '2';
else *lptr++ = '1';
tstr = fstr;
while (*tstr != '\0') *lptr++ = *tstr++;
*lptr = '\0';
at (7,1,"Is this string to be ended by <RETURN>? ");
ch=response(echo);
if (ch == YES) {
*lptr = '\r'; *++lptr = '\0';
}
if (fnum > 11) {
start = 53; /* Set 2nd column */
fnum -= 11; /* Adjust to 1-11 */
}
else start = 31;
for (i=start; i<start+20; i++) at (fnum+10, i, " ");
at (fnum+10,start,"");
prfkeystr(fstr);
}
} while (fnum != 0);
return(YES);
}
fkeystatus() /* Display Status of Function Keys */
{
char ch;
int i;
cmnd('$'); /* Graphics ON */
at (10,30,"B"); at (10,74,"C"); /* Draw Box for Fct Keys */
at (22,30,"A"); at (22,74,"D");
for (i=1; i<=11; i++) {
at (i+10,30,"J");
at (i+10,52,"J");
at (i+10,74,"J");
}
for (i=31; i<52; i++) {
at (10,i,"K");
at (22,i,"K");
}
at (10,52,"N"); at (22,52,"O");
for (i=53; i<=73; i++) {
at (10,i,"K");
at (22,i,"K");
}
cmnd('%'); /* Graphics OFF */
/* Store current function key definitions in box */
for (i=1; i<=11; i++) {
at (i+10,31,"");
prfkeystr(fkeybuffer+(i-1)*20+1);
at (i+10,53,"");
prfkeystr(fkeybuffer+(i+10)*20+1);
}
/* Box titles and numbers along left side of box */
at (9,31,"Function Key"); at (9,53,"Shifted Function Key");
for (i=1; i<=11; i++) {
if (i < 10) at (i+10,28,"");
else at (i+10,27,"");
printf("%d",i);
}
}
setclick() /* This routine turn the key click on or off */
{
char ch;
clreos(); /* Clear Display Area */
at (18, 1,"Do you want the Key Click ON? ");
ch=response(echo);
at (20,10,"Key Click is ");
if (ch==YES) {
puts("ON");
return(clkon);
}
else {
puts("OFF");
return(clkoff);
}
}
setscroll() /* This routine turn the smooth scroll on or off */
{
char ch;
clreos(); /* Clear Display Area */
at (18, 1,"Do you want Smooth Scrolling ON? ");
ch=response(echo);
at (20,10,"Smooth Scrolling is ");
if (ch==YES) {
puts("ON");
return(sscroll);
}
else {
puts("OFF");
return(hscroll);
}
}
settabs() /* This routine allows the user to specify tab locations */
{
char *lptr, ch;
int i;
clreos(); /* Clear Display Area */
lptr = tabbuffer; /* Pt to first char of tab buffer */
for (i=0;i<tablen;i++) *lptr++=' '; /* Clear tab buffer */
i = 1; lptr = tabbuffer; /* Set char pos and ptr to 1st char */
at (18,1,"Do you wish to Enable Tabs? ");
ch=response(echo);
if (ch != YES) return(NO);
at (18,1,"Set Tab Stops -- . Sets, <SPACE BAR> Clears");
at (19,1,"Left and Right Arrow Keys move Cursor Left and Right");
at (20,1,"<RETURN> terminates entry of tab stops");
at (24,1,"");
do {
ch=response(noecho); /* Get char */
switch (ch) {
case CR : break;
case ' ' :
case '.' : *lptr++ = ch; /* Store Char */
putchar(ch); /* Echo Char */
i++; /* Incr col counter */
if (i > tablen) { /* Beyond EOL */
i--; lptr--;
putchar(CTRLH);
}
break;
case CTRLH : if (i == 1) putchar(BEL);
else { /* left arrow */
i--; lptr--;
putchar(ch);
}
break;
case CTRLL : if (i == tablen) putchar(BEL);
else { /* right arrow */
i++; lptr++;
putchar(ch);
}
break;
default : putchar(BEL); break;
}
} while (ch != CR);
return(YES);
}
setuline() /* This routine turn the smooth scroll on or off */
{
char ch;
clreos(); /* Clear Display Area */
at (18, 1,"Do you want the User Line Displayed? ");
ch=response(echo);
at (19,10,"User Line Display is ");
if (ch==YES) {
printf("ON\nOld User Line was --\n%s",ulbuffer);
printf("\nInput Your User Line --\n");
inline(ulbuffer,ullen);
return(YES);
}
else {
puts("OFF");
return(NO);
}
}
setvideo() /* This routine selects the screen video attribute */
{
char ch;
clreos(); /* Clear Display Area */
at (18, 1,"Do you want Reverse Video? ");
ch=response(echo);
if (ch==YES) {
at (20,10,"Reverse Video is ON");
return(vidrev);
}
else {
at (20,10,"Normal Video is ON");
return(vidnorm);
}
}
readwrite(filename) /* Read or Write a Data File or Write CONFIG.COM */
char *filename;
{
char ch, tname[20], oname[20], *bptr;
int fd, ioblen;
oname[1] = '\0'; /* clear output file name */
/* Loop until exit command specified */
do {
/* Input valid command */
do {
clreos(); /* Clear to end of screen */
at (18,1,"Select one of the following --");
at (19,1,"0 - Exit 1 - Read a File");
at (20,1,"2 - Write a File 3 - Write CONFIG.COM");
at (22,1,"Selection? "); ch=response(echo);
} while ((ch < '0') | (ch > '3'));
/* Get file name for Read or Write of a data file */
if ((ch == '1') | (ch == '2')) {
at (23,1,"Name of File (<RETURN> = ");
printf("%s)? ", filename);
dots(20); inline (tname,20);
bptr = tname; /* Pt to first char of file name */
while (*bptr != '\0') {
*bptr = toupper(*bptr);
bptr++;
}
if (tname[0] == '\0') strcpy (oname,filename);
else strcpy (oname,tname);
}
switch (ch) {
case CR : ch = '0'; break; /* <CR> = Done */
case ' ' : ch = '0'; break; /* <SP> = Done */
case '0' : break;
case '1' : readfile (oname); break;
case '2' : writefile (oname); break;
case '3' : printf("\nWriting CONFIG.COM to Disk");
fd = fcreat(configfile, iobuf);
ioblen = endext() - 0x100;
bptr = 0x100;
writebuffer (bptr, ioblen);
fflush (iobuf);
fclose (iobuf);
break;
}
} while (ch != '0');
}
cmderror(ch1) /* Invalid Command Received -- Print error message */
char ch1;
{
clreos(); /* Clear to EOS from 18,1 */
at (18,1,"Invalid Command: ");
putchar(ch1);
}
clreos() /* Clear to End of Screen (EOS) for SETUP */
{
at (18, 1,""); /* Position Cursor */
printf("%c%c",ESC,'Y');
sleep(1); /* Brief Delay */
}
/********************************************************************
Routines to Read and Write Configuration Data
********************************************************************/
readfile(name) /* Read Indicated File */
char *name;
{
int fd, i, type;
char newfile[20], filename[20], *ptr;
ptr = name; /* Copy passed name into FILENAME */
type = FALSE;
i = 0;
while (*ptr != '\0') {
if (*ptr == '.') type = TRUE; /* Note if file type given */
filename[i++] = *ptr++;
}
filename[i] = '\0';
if (!type) strcat (filename,configtype); /* Set default file type */
fd = fopen (filename, iobuf); /* Try to open file */
if (fd == ERROR) { /* If failure, try to open file on A: */
newfile[0] = '\0';
strcat (newfile,"A:");
strcat (newfile,filename);
fd = fopen (newfile, iobuf);
if (fd == ERROR) { /* If this fails, give error */
printf("File %s not found", filename);
exit (FALSE);
}
}
/* Read in parameters from disk file */
cursor = getc(iobuf);
if (('0'<=cursor) & (cursor<='4')) /* OK */;
else ferror();
fkeys = getc(iobuf);
if ((fkeys == YES) | (fkeys == NO)) /* OK */;
else ferror();
click = getc(iobuf);
if ((click == clkon) | (click == clkoff)) /* OK */;
else ferror();
scroll = getc(iobuf);
if ((scroll == sscroll) | (scroll == hscroll)) /* OK */;
else ferror();
tabs = getc(iobuf);
if ((tabs == YES) | (tabs == NO)) /* OK */;
else ferror();
uline = getc(iobuf);
if ((uline == YES) | (uline == NO)) /* OK */;
else ferror();
video = getc(iobuf);
if ((video == vidrev) | (video == vidnorm)) /* OK */;
else ferror();
getc(iobuf); getc(iobuf); /* Ignore <CR> <LF> */
if (uline == YES) readbuffer (ulbuffer, ullen);
if (tabs == YES) readbuffer (tabbuffer, tablen);
if (fkeys == YES) readbuffer (fkeybuffer, fkeylen);
fclose(iobuf);
}
readbuffer (bufname, buflen) /* Read into the buffer pted to by bufname */
char *bufname;
int buflen;
{
int i, inch;
char *bptr;
bptr = bufname;
for (i=1;i<=buflen;i++) {
inch = getc(iobuf);
if (inch == ERROR) ferror();
*bptr++ = inch;
}
getc(iobuf); getc(iobuf); /* Flush <CR> <LF> */
}
writefile (name) /* Write the file to disk */
char *name;
{
int fd, i, type;
char filename[20], *ptr;
ptr = name; /* Copy passed name into FILENAME */
type = FALSE;
i = 0;
while (*ptr != '\0') {
if (*ptr == '.') type = TRUE; /* Note if file type given */
filename[i++] = *ptr++;
}
filename[i] = '\0';
if (!type) strcat (filename,configtype); /* Set default file type */
fd = fcreat(filename, iobuf); /* Create new file */
if (fd == ERROR) {
puts("\nDisk Full");
ferror();
}
prfkeystr(ptr) /* Print function key string with <ESCAPE> char processing */
char *ptr;
{
while (*ptr != '\0') if (*ptr != ESC) putchar(*ptr++);
else {
putchar('$');
ptr++;
}
}
ferror() /* File Format Error Encountered */
{
puts("\nAborting on Fatal Error in Data File");
exit (FALSE);
}
dots(ndots) /* Print Dots */
int ndots;
{
int i;
for (i=1;i<=ndots;i++) putchar('.');
for (i=1;i<=ndots;i++) putchar(CTRLH);
}
inline(buffer,nchars) /* Input a line from user into BUFFER */
char *buffer;
int nchars;
{
int i, j;
char *lptr, ch;
i = 1; lptr = buffer; /* Init count and ptr */
do {
ch=bios(3); /* Get char with no echo and no caps */
switch (ch) {
case CR : *lptr = '\0'; /* End String */
break;
case RUBOUT :
case CTRLH : if (i == 1) putchar(BEL);
else {
i--; lptr--;
rubout();
}
break;
case CTRLU :
case CTRLX : for (j=i;j>1;j--) {
rubout();
}
i = 1; lptr = buffer;
break;
default : if (i == nchars) putchar(BEL);
else {
*lptr++ = ch;
i++;
if (ch != ESC) putchar(ch);
else putchar('$');
}
break;
}
} while (ch != CR);
}