#include "lp.cdefs.h"
#include <sys/param.h>
#include <sys/mount.h>
#include <sys/stat.h>
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
#include <stdint.h>
#include <signal.h>
#include <stdarg.h>
#include <stdckdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <syslog.h>
#include <unistd.h>
#include "lp.h"
#include "lp.local.h"
#include "ctlinfo.h"
#include "extern.h"
#include "pathnames.h"
#define digit(ch) ((ch) >= '0' && (ch) <= '9')
#define SPL_BUFSIZ BUFSIZ
static char dfname[NAME_MAX];
static size_t minfree;
static char tfname[NAME_MAX];
static void ack(struct printer *_pp);
static void nak(struct printer *_pp);
static int chksize(size_t _size);
static void frecverr(const char *_msg, ...) __printf0like(1, 2);
static int noresponse(void);
static void rcleanup(int _signo);
static void read_minfree(void);
static int readfile(struct printer *_pp, char *_file, size_t _size);
static int readjob(struct printer *_pp);
void
recvjob(const char *printer)
{
struct stat stb;
int status;
struct printer myprinter, *pp = &myprinter;
init_printer(pp);
status = getprintcap(printer, pp);
switch (status) {
case PCAPERR_OSERR:
frecverr("cannot open printer description file");
break;
case PCAPERR_NOTFOUND:
frecverr("unknown printer %s", printer);
break;
case PCAPERR_TCLOOP:
fatal(pp, "potential reference loop detected in printcap file");
default:
break;
}
(void) close(STDERR_FILENO);
if (open(pp->log_file, O_WRONLY|O_APPEND, 0664) < 0) {
syslog(LOG_ERR, "%s: %m", pp->log_file);
(void) open(_PATH_DEVNULL, O_WRONLY);
}
if (chdir(pp->spool_dir) < 0)
frecverr("%s: chdir(%s): %s", pp->printer, pp->spool_dir,
strerror(errno));
if (stat(pp->lock_file, &stb) == 0) {
if (stb.st_mode & 010) {
putchar('\1');
exit(1);
}
} else if (stat(pp->spool_dir, &stb) < 0)
frecverr("%s: stat(%s): %s", pp->printer, pp->spool_dir,
strerror(errno));
read_minfree();
signal(SIGTERM, rcleanup);
signal(SIGPIPE, rcleanup);
if (readjob(pp))
printjob(pp);
}
static int
readjob(struct printer *pp)
{
ssize_t rlen;
size_t len, size;
int cfcnt, dfcnt, curd0, curd2, curn, n;
char *cp, *clastp, *errmsg, *sep;
char givenid[32], givenhost[MAXHOSTNAMELEN];
tfname[0] = dfname[0] = '\0';
ack(pp);
cfcnt = 0;
dfcnt = 0;
curd0 = 'd';
curd2 = 'A';
curn = -1;
for (;;) {
cp = line;
clastp = line + sizeof(line) - 1;
do {
rlen = read(STDOUT_FILENO, cp, 1);
if (rlen != 1) {
if (rlen < 0) {
frecverr("%s: lost connection",
pp->printer);
}
return (cfcnt);
}
} while (*cp++ != '\n' && cp <= clastp);
if (cp > clastp) {
frecverr("%s: readjob overflow", pp->printer);
}
*--cp = '\0';
cp = line;
switch (*cp++) {
case '\1':
rcleanup(0);
continue;
case '\2':
if (tfname[0] != '\0') {
break;
}
for (size = 0; digit(*cp); cp++) {
if (ckd_mul(&size, size, 10) ||
ckd_add(&size, size, *cp - '0'))
break;
}
if (*cp++ != ' ')
break;
if (cp[0] != 'c' || cp[1] != 'f' || cp[2] != 'A' ||
!digit(cp[3]) || !digit(cp[4]) || !digit(cp[5]))
break;
n = (cp[3] - '0') * 100 + (cp[4] - '0') * 10 +
cp[5] - '0';
if (curn == -1)
curn = n;
else if (curn != n)
break;
len = snprintf(tfname, sizeof(tfname), "%.6s%s", cp,
from_host);
if (len >= sizeof(tfname))
break;
tfname[0] = 't';
if (!chksize(size)) {
nak(pp);
continue;
}
if (!readfile(pp, tfname, size)) {
rcleanup(0);
continue;
}
errmsg = ctl_renametf(pp->printer, tfname);
tfname[0] = '\0';
if (errmsg != NULL) {
frecverr("%s: %s", pp->printer, errmsg);
}
cfcnt++;
continue;
case '\3':
if (curd0 > 'z') {
break;
}
*givenid = '\0';
*givenhost = '\0';
for (size = 0; digit(*cp); cp++) {
if (ckd_mul(&size, size, 10) ||
ckd_add(&size, size, *cp - '0'))
break;
}
if (*cp++ != ' ')
break;
if (cp[0] != curd0 || cp[1] != 'f' || cp[2] != curd2 ||
!digit(cp[3]) || !digit(cp[4]) || !digit(cp[5]))
break;
n = (cp[3] - '0') * 100 + (cp[4] - '0') * 10 +
cp[5] - '0';
if (curn == -1)
curn = n;
else if (curn != n)
break;
len = snprintf(dfname, sizeof(dfname), "%.6s%s", cp,
from_host);
if (len >= sizeof(dfname))
break;
if (!chksize(size)) {
nak(pp);
continue;
}
switch (curd2++) {
case 'Z':
curd2 = 'a';
break;
case 'z':
curd0++;
curd2 = 'A';
break;
}
dfcnt++;
trstat_init(pp, dfname, dfcnt);
if (!readfile(pp, dfname, size)) {
rcleanup(0);
continue;
}
trstat_write(pp, TR_RECVING, size, givenid,
from_host, givenhost);
continue;
}
frecverr("protocol screwup: %s", line);
}
}
static int
readfile(struct printer *pp, char *file, size_t size)
{
char buf[SPL_BUFSIZ];
ssize_t rlen, wlen;
int err, f0, fd, j;
fd = open(file, O_CREAT|O_EXCL|O_WRONLY, FILMOD);
if (fd < 0) {
f0 = file[0];
file[0] = '\0';
frecverr("%s: readfile: error on open(%c%s): %s",
pp->printer, f0, file + 1, strerror(errno));
}
ack(pp);
while (size > 0) {
rlen = read(STDOUT_FILENO, buf, MIN(SPL_BUFSIZ, size));
if (rlen < 0 && errno == EINTR)
continue;
if (rlen <= 0) {
frecverr("%s: lost connection", pp->printer);
}
size -= rlen;
while (rlen > 0) {
wlen = write(fd, buf, rlen);
if (wlen < 0 && errno == EINTR)
continue;
if (wlen <= 0) {
frecverr("%s: write error on write(%s)",
pp->printer, file);
}
rlen -= wlen;
}
}
if (close(fd) != 0) {
frecverr("%s: write error on close(%s)", pp->printer, file);
}
if (noresponse()) {
(void) unlink(file);
return (0);
}
ack(pp);
return (1);
}
static int
noresponse(void)
{
char resp;
if (read(STDOUT_FILENO, &resp, 1) != 1) {
frecverr("lost connection in noresponse()");
}
if (resp == '\0')
return (0);
return (1);
}
static int
chksize(size_t size)
{
struct statfs sfb;
size_t avail;
if (statfs(".", &sfb) < 0) {
syslog(LOG_ERR, "%s: %m", "statfs(\".\")");
return (1);
}
if (ckd_mul(&avail, sfb.f_bavail, (sfb.f_bsize / 512)))
return (1);
if (avail <= minfree)
return (0);
if (avail - minfree <= size / 512)
return (0);
return (1);
}
static void
read_minfree(void)
{
FILE *fp;
minfree = 0;
if ((fp = fopen("minfree", "r")) != NULL) {
if (fscanf(fp, "%zu", &minfree) != 1)
minfree = 0;
fclose(fp);
}
if (ckd_mul(&minfree, minfree, 2))
minfree = SIZE_MAX;
}
static void
rcleanup(int signo __unused)
{
if (tfname[0] && strchr(tfname, '/') == NULL)
(void) unlink(tfname);
if (dfname[0] && strchr(dfname, '/') == NULL) {
do {
do
(void) unlink(dfname);
while (dfname[2]-- != 'A');
dfname[2] = 'z';
} while (dfname[0]-- != 'd');
}
dfname[0] = '\0';
}
static void
frecverr(const char *msg, ...)
{
va_list ap;
va_start(ap, msg);
syslog(LOG_ERR, "Error receiving job from %s:", from_host);
vsyslog(LOG_ERR, msg, ap);
va_end(ap);
rcleanup(0);
sleep(2);
putchar('\1');
exit(1);
}
static void
ack(struct printer *pp)
{
if (write(STDOUT_FILENO, "\0", 1) != 1)
frecverr("%s: write error on ack", pp->printer);
}
static void
nak(struct printer *pp)
{
if (write(STDOUT_FILENO, "\2", 1) != 1)
frecverr("%s: write error on nak", pp->printer);
}