#include "stdio.h"
#include "string.h"
#include "errno.h"
#include "sys/types.h"
#include "stdlib.h"
#include <syslog.h>
#include "lp.h"
#include "printers.h"
extern struct {
char *v;
short len,
okremote;
} prtrheadings[];
PRINTER *
getprinter(char *name)
{
static long lastdir = -1;
PRINTER *prp;
char buf[BUFSIZ];
short daisy;
int fld;
int fd;
FALERT *pa;
register char * p;
register char ** pp;
register char *** ppp;
register char * path;
int isNameAll;
if (!name || !*name) {
errno = EINVAL;
return (0);
}
syslog(LOG_DEBUG, "getprinter(%s)", name ? name : "");
isNameAll = STREQU(NAME_ALL, name);
for (; ; ) {
if (isNameAll) {
if (!(name = next_dir(Lp_A_Printers, &lastdir)))
return (0);
} else
lastdir = -1;
path = getprinterfile(name, CONFIGFILE);
if (!path) {
if (isNameAll)
Free(name);
return (0);
}
if ((fd = open_locked(path, "r", 0)) < 0) {
Free(path);
if (!isNameAll)
return(0);
else
Free(name);
}
else
break;
}
Free (path);
prp = calloc(sizeof (*prp), 1);
prp->name = Strdup(name);
if (isNameAll)
Free(name);
prp->printer_types = getlist(NAME_UNKNOWN, LP_WS, LP_SEP);
prp->input_types = getlist(NAME_SIMPLE, LP_WS, LP_SEP);
#if defined(CAN_DO_MODULES)
prp->modules = getlist(NAME_DEFAULT, LP_WS, LP_SEP);
#endif
errno = 0;
while (fdgets(buf, BUFSIZ, fd) != NULL) {
buf[strlen(buf) - 1] = 0;
for (fld = 0; fld < PR_MAX; fld++)
if (
prtrheadings[fld].v
&& prtrheadings[fld].len
&& STRNEQU(
buf,
prtrheadings[fld].v,
prtrheadings[fld].len
)
) {
p = buf + prtrheadings[fld].len;
while (*p && *p == ' ')
p++;
break;
}
if (fld >= PR_MAX)
continue;
switch (fld) {
case PR_BAN:
if ((pp = getlist(p, LP_WS, ":"))) {
if (pp[0] != NULL) {
if (strcmp(pp[0], NAME_OPTIONAL) == 0)
prp->banner = BAN_OPTIONAL;
else if (strcmp(pp[0], NAME_OFF) == 0)
prp->banner = BAN_NEVER;
else if (strcmp(pp[0], NAME_ON) == 0)
prp->banner = BAN_ALWAYS;
else
prp->banner = BAN_ALWAYS;
}
if (pp[1] && CS_STREQU(pp[1], NAME_ALWAYS))
prp->banner |= BAN_ALWAYS;
freelist (pp);
}
break;
case PR_LOGIN:
prp->login = LOG_IN;
break;
case PR_CPI:
prp->cpi = getcpi(p);
break;
case PR_LPI:
prp->lpi = getsdn(p);
break;
case PR_LEN:
prp->plen = getsdn(p);
break;
case PR_WIDTH:
prp->pwid = getsdn(p);
break;
case PR_CS:
ppp = &(prp->char_sets);
goto CharStarStar;
case PR_ITYPES:
ppp = &(prp->input_types);
CharStarStar: if (*ppp)
freelist (*ppp);
*ppp = getlist(p, LP_WS, LP_SEP);
break;
case PR_DEV:
pp = &(prp->device);
goto CharStar;
case PR_DIAL:
pp = &(prp->dial_info);
goto CharStar;
case PR_RECOV:
pp = &(prp->fault_rec);
goto CharStar;
case PR_INTFC:
pp = &(prp->interface);
goto CharStar;
case PR_PTYPE:
ppp = &(prp->printer_types);
goto CharStarStar;
case PR_REMOTE:
pp = &(prp->remote);
goto CharStar;
case PR_SPEED:
pp = &(prp->speed);
goto CharStar;
case PR_STTY:
pp = &(prp->stty);
CharStar: if (*pp)
Free (*pp);
*pp = Strdup(p);
break;
#if defined(CAN_DO_MODULES)
case PR_MODULES:
ppp = &(prp->modules);
goto CharStarStar;
#endif
case PR_OPTIONS:
ppp = &(prp->options);
goto CharStarStar;
break;
case PR_PPD:
{
pp = &(prp->ppd);
goto CharStar;
}
}
}
if (errno != 0) {
int save_errno = errno;
freeprinter (prp);
close(fd);
errno = save_errno;
return (0);
}
close(fd);
if (!(path = getprinterfile(prp->name, COMMENTFILE)))
return (0);
if (!(prp->description = loadstring(path)) && errno != ENOENT) {
Free (path);
freeprinter (prp);
return (0);
}
Free (path);
if (!(pa = getalert(Lp_A_Printers, prp->name))) {
if (
errno != ENOENT
&& (
errno != EACCES
|| !getpid()
|| STREQU(getname(), LPUSER)
)
) {
freeprinter (prp);
return (0);
}
} else
prp->fault_alert = *pa;
if (!okprinter(prp->name, prp, 0)) {
freeprinter (prp);
errno = EBADF;
return (0);
}
if (!prp->printer_types)
prp->printer_types = getlist(NAME_UNKNOWN, LP_WS, LP_SEP);
if (
lenlist(prp->printer_types) > 1
&& prp->input_types
&& (
lenlist(prp->input_types) > 1
|| !STREQU(NAME_SIMPLE, *prp->input_types)
)
) {
freeprinter (prp);
badprinter = BAD_ITYPES;
errno = EBADF;
return (0);
}
if (
lenlist(prp->printer_types) > 1
&& searchlist(NAME_UNKNOWN, prp->printer_types)
) {
freeprinter (prp);
badprinter = BAD_PTYPES;
errno = EBADF;
return (0);
}
prp->daisy = -1;
for (pp = prp->printer_types; *pp; pp++) {
tidbit (*pp, "daisy", &daisy);
if (daisy == -1)
daisy = 0;
if (prp->daisy == -1)
prp->daisy = daisy;
else if (prp->daisy != daisy) {
freeprinter (prp);
badprinter = BAD_DAISY;
errno = EBADF;
return (0);
}
}
prp->printer_type = Strdup(*prp->printer_types);
return (prp);
}