xref: /onnv-gate/usr/src/cmd/lp/lib/papi/job.c (revision 2660)
10Sstevel@tonic-gate /*
20Sstevel@tonic-gate  * CDDL HEADER START
30Sstevel@tonic-gate  *
40Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
51676Sjpk  * Common Development and Distribution License (the "License").
61676Sjpk  * You may not use this file except in compliance with the License.
70Sstevel@tonic-gate  *
80Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
90Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
100Sstevel@tonic-gate  * See the License for the specific language governing permissions
110Sstevel@tonic-gate  * and limitations under the License.
120Sstevel@tonic-gate  *
130Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
140Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
150Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
160Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
170Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
180Sstevel@tonic-gate  *
190Sstevel@tonic-gate  * CDDL HEADER END
200Sstevel@tonic-gate  */
210Sstevel@tonic-gate /*
221676Sjpk  * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
230Sstevel@tonic-gate  * Use is subject to license terms.
240Sstevel@tonic-gate  */
250Sstevel@tonic-gate 
260Sstevel@tonic-gate #pragma ident	"%Z%%M%	%I%	%E% SMI"
270Sstevel@tonic-gate 
280Sstevel@tonic-gate /*LINTLIBRARY*/
290Sstevel@tonic-gate 
300Sstevel@tonic-gate #include <stdlib.h>
310Sstevel@tonic-gate #include <string.h>
320Sstevel@tonic-gate #include <unistd.h>
330Sstevel@tonic-gate #include <libintl.h>
340Sstevel@tonic-gate #include <pwd.h>
350Sstevel@tonic-gate #include <sys/stat.h>
360Sstevel@tonic-gate #include <papi_impl.h>
370Sstevel@tonic-gate 
382264Sjacobs 
392264Sjacobs /*
402264Sjacobs  * for an older application that may have been linked with a pre-v1.0
412264Sjacobs  * PAPI implementation.
422264Sjacobs  */
432264Sjacobs papi_status_t
442264Sjacobs papiAttributeListAdd(papi_attribute_t ***attrs, int flags, char *name,
452264Sjacobs 		papi_attribute_value_type_t type, papi_attribute_value_t *value)
462264Sjacobs {
472264Sjacobs 	return (papiAttributeListAddValue(attrs, flags, name, type, value));
482264Sjacobs }
492264Sjacobs 
500Sstevel@tonic-gate #ifdef LP_USE_PAPI_ATTR
512264Sjacobs static papi_status_t psm_modifyAttrsFile(papi_attribute_t **attrs, char *file);
522264Sjacobs static papi_status_t psm_modifyAttrsList(char *file, papi_attribute_t **attrs,
532264Sjacobs 					papi_attribute_t ***newAttrs);
540Sstevel@tonic-gate #endif
550Sstevel@tonic-gate 
560Sstevel@tonic-gate 
570Sstevel@tonic-gate void
580Sstevel@tonic-gate papiJobFree(papi_job_t job)
590Sstevel@tonic-gate {
600Sstevel@tonic-gate 	job_t *tmp = (job_t *)job;
610Sstevel@tonic-gate 
620Sstevel@tonic-gate 	if (tmp != NULL) {
630Sstevel@tonic-gate 		papiAttributeListFree(tmp->attributes);
640Sstevel@tonic-gate 		free(tmp);
650Sstevel@tonic-gate 	}
660Sstevel@tonic-gate }
670Sstevel@tonic-gate 
680Sstevel@tonic-gate void
690Sstevel@tonic-gate papiJobListFree(papi_job_t *jobs)
700Sstevel@tonic-gate {
710Sstevel@tonic-gate 	if (jobs != NULL) {
720Sstevel@tonic-gate 		int i;
730Sstevel@tonic-gate 
740Sstevel@tonic-gate 		for (i = 0; jobs[i] != NULL; i++) {
750Sstevel@tonic-gate 			papiJobFree(jobs[i]);
760Sstevel@tonic-gate 		}
770Sstevel@tonic-gate 		free(jobs);
780Sstevel@tonic-gate 	}
790Sstevel@tonic-gate }
800Sstevel@tonic-gate 
810Sstevel@tonic-gate papi_attribute_t **
820Sstevel@tonic-gate papiJobGetAttributeList(papi_job_t job)
830Sstevel@tonic-gate {
840Sstevel@tonic-gate 	job_t *tmp = (job_t *)job;
850Sstevel@tonic-gate 
860Sstevel@tonic-gate 	if (tmp != NULL)
870Sstevel@tonic-gate 		return (tmp->attributes);
880Sstevel@tonic-gate 
890Sstevel@tonic-gate 	return (NULL);
900Sstevel@tonic-gate }
910Sstevel@tonic-gate 
920Sstevel@tonic-gate char *
930Sstevel@tonic-gate papiJobGetPrinterName(papi_job_t job)
940Sstevel@tonic-gate {
950Sstevel@tonic-gate 	job_t *tmp = (job_t *)job;
960Sstevel@tonic-gate 	char *result = NULL;
970Sstevel@tonic-gate 
980Sstevel@tonic-gate 	if (tmp != NULL)
990Sstevel@tonic-gate 		papiAttributeListGetString(tmp->attributes, NULL,
1000Sstevel@tonic-gate 					"printer-name", &result);
1010Sstevel@tonic-gate 
1020Sstevel@tonic-gate 	return (result);
1030Sstevel@tonic-gate }
1040Sstevel@tonic-gate 
1050Sstevel@tonic-gate int32_t
1060Sstevel@tonic-gate papiJobGetId(papi_job_t job)
1070Sstevel@tonic-gate {
1080Sstevel@tonic-gate 	job_t *tmp = (job_t *)job;
1090Sstevel@tonic-gate 	int result = -1;
1100Sstevel@tonic-gate 
1110Sstevel@tonic-gate 	if (tmp != NULL)
1120Sstevel@tonic-gate 		papiAttributeListGetInteger(tmp->attributes, NULL, "job-id",
1130Sstevel@tonic-gate 					&result);
1140Sstevel@tonic-gate 
1150Sstevel@tonic-gate 	return (result);
1160Sstevel@tonic-gate }
1170Sstevel@tonic-gate 
1180Sstevel@tonic-gate static REQUEST *
1190Sstevel@tonic-gate create_request(papi_service_t svc, char *printer, papi_attribute_t **attributes)
1200Sstevel@tonic-gate {
1210Sstevel@tonic-gate 	static REQUEST r;
1220Sstevel@tonic-gate 
1230Sstevel@tonic-gate 	memset(&r, 0, sizeof (r));
1240Sstevel@tonic-gate 	r.priority = -1;
1250Sstevel@tonic-gate 	r.destination = printer_name_from_uri_id(printer, -1);
1260Sstevel@tonic-gate 	job_attributes_to_lpsched_request(svc, &r, attributes);
1270Sstevel@tonic-gate 
1280Sstevel@tonic-gate 	return (&r);
1290Sstevel@tonic-gate }
1300Sstevel@tonic-gate 
1310Sstevel@tonic-gate static papi_status_t
1320Sstevel@tonic-gate authorized(service_t *svc, int32_t id)
1330Sstevel@tonic-gate {
1340Sstevel@tonic-gate 	papi_status_t result = PAPI_NOT_AUTHORIZED;	/* assume the worst */
1350Sstevel@tonic-gate 	char file[32];
1360Sstevel@tonic-gate 	REQUEST *r;
1370Sstevel@tonic-gate 
1380Sstevel@tonic-gate 	snprintf(file, sizeof (file), "%d-0", id);
1390Sstevel@tonic-gate 	if ((r = getrequest(file)) != NULL) {
1400Sstevel@tonic-gate 		uid_t uid = getuid();
1410Sstevel@tonic-gate 		struct passwd *pw = NULL;
1420Sstevel@tonic-gate 		char *user = "intruder";	/* assume an intruder */
1430Sstevel@tonic-gate 
1440Sstevel@tonic-gate 		if ((pw = getpwuid(uid)) != NULL)
1450Sstevel@tonic-gate 			user = pw->pw_name;	/* use the process owner */
1460Sstevel@tonic-gate 
1470Sstevel@tonic-gate 		if ((uid == 0) || (uid == 71)) { /* root/lp can forge this */
1480Sstevel@tonic-gate 			papi_status_t s;
1490Sstevel@tonic-gate 			s = papiAttributeListGetString(svc->attributes, NULL,
1500Sstevel@tonic-gate 					"user-name", &user);
1510Sstevel@tonic-gate 			if (s != PAPI_OK)	/* true root/lp are almighty */
1520Sstevel@tonic-gate 				result = PAPI_OK;
1530Sstevel@tonic-gate 		}
1540Sstevel@tonic-gate 
1550Sstevel@tonic-gate 		if ((result != PAPI_OK) && (strcmp(user, r->user) == 0))
1560Sstevel@tonic-gate 			result = PAPI_OK;
1570Sstevel@tonic-gate 
1580Sstevel@tonic-gate 		freerequest(r);
1590Sstevel@tonic-gate 	} else
1600Sstevel@tonic-gate 		result = PAPI_NOT_FOUND;
1610Sstevel@tonic-gate 
1620Sstevel@tonic-gate 	return (result);
1630Sstevel@tonic-gate }
1640Sstevel@tonic-gate 
1650Sstevel@tonic-gate static papi_status_t
1662264Sjacobs copy_file(char *from, char *to)
1670Sstevel@tonic-gate {
1680Sstevel@tonic-gate 	int ifd, ofd;
1690Sstevel@tonic-gate 	char buf[BUFSIZ];
1700Sstevel@tonic-gate 	int rc;
1710Sstevel@tonic-gate 
1720Sstevel@tonic-gate 	if ((ifd = open(from, O_RDONLY)) < 0)
1730Sstevel@tonic-gate 		return (PAPI_DOCUMENT_ACCESS_ERROR);
1740Sstevel@tonic-gate 
1750Sstevel@tonic-gate 	if ((ofd = open(to, O_WRONLY)) < 0) {
1760Sstevel@tonic-gate 		close(ifd);
1770Sstevel@tonic-gate 		return (PAPI_NOT_POSSIBLE);
1780Sstevel@tonic-gate 	}
1790Sstevel@tonic-gate 
1800Sstevel@tonic-gate 	while ((rc = read(ifd, buf, sizeof (buf))) > 0)
1810Sstevel@tonic-gate 		write(ofd, buf, rc);
1820Sstevel@tonic-gate 
1830Sstevel@tonic-gate 	close(ifd);
1840Sstevel@tonic-gate 	close(ofd);
1850Sstevel@tonic-gate 
1860Sstevel@tonic-gate 	return (PAPI_OK);
1870Sstevel@tonic-gate }
1880Sstevel@tonic-gate 
1890Sstevel@tonic-gate 
1900Sstevel@tonic-gate #ifdef LP_USE_PAPI_ATTR
1910Sstevel@tonic-gate /*
1920Sstevel@tonic-gate  * *****************************************************************************
1930Sstevel@tonic-gate  *
1940Sstevel@tonic-gate  * Description: Create a file containing all the attributes in the attribute
1950Sstevel@tonic-gate  *              list passed to this function.
1960Sstevel@tonic-gate  *              This file is then passed through lpsched and given to either
1970Sstevel@tonic-gate  *              a slow-filter or to the printer's interface script to process
1980Sstevel@tonic-gate  *              the attributes.
1990Sstevel@tonic-gate  *
2000Sstevel@tonic-gate  * Parameters:  attrs - list of attributes and their values
2010Sstevel@tonic-gate  *              file  - file pathname to create and put the attributes into.
2020Sstevel@tonic-gate  *
2030Sstevel@tonic-gate  * *****************************************************************************
2040Sstevel@tonic-gate  */
2050Sstevel@tonic-gate 
2060Sstevel@tonic-gate static papi_status_t
2072264Sjacobs psm_copy_attrsToFile(papi_attribute_t **attrs, char *file)
2080Sstevel@tonic-gate 
2090Sstevel@tonic-gate {
2100Sstevel@tonic-gate 	papi_status_t result = PAPI_OK;
2112264Sjacobs 
2122264Sjacobs 	if ((attrs != NULL) && (*attrs != NULL)) {
2132264Sjacobs 		FILE *out = NULL;
2140Sstevel@tonic-gate 
2152264Sjacobs 		if ((out = fopen(file, "w")) != NULL) {
2162264Sjacobs 			papiAttributeListPrint(out, attrs, "");
2170Sstevel@tonic-gate 			fclose(out);
2182264Sjacobs 		} else {
2190Sstevel@tonic-gate 			result = PAPI_NOT_POSSIBLE;
2200Sstevel@tonic-gate 		}
2210Sstevel@tonic-gate 	}
2220Sstevel@tonic-gate 
2230Sstevel@tonic-gate 	return (result);
2240Sstevel@tonic-gate } /* psm_copy_attrsToFile */
2250Sstevel@tonic-gate 
2260Sstevel@tonic-gate 
2270Sstevel@tonic-gate /*
2280Sstevel@tonic-gate  * *****************************************************************************
2290Sstevel@tonic-gate  *
2300Sstevel@tonic-gate  * Description: Modify the given attribute 'file' with the attributes from the
2310Sstevel@tonic-gate  *              'attrs' list. Attributes already in the file will be replaced
2320Sstevel@tonic-gate  *              with the new value. New attributes will be added into the file.
2330Sstevel@tonic-gate  *
2340Sstevel@tonic-gate  * Parameters:  attrs - list of attributes and their values
2350Sstevel@tonic-gate  *              file  - file pathname to create and put the attributes into.
2360Sstevel@tonic-gate  *
2370Sstevel@tonic-gate  * *****************************************************************************
2380Sstevel@tonic-gate  */
2390Sstevel@tonic-gate 
2400Sstevel@tonic-gate static papi_status_t
2412264Sjacobs psm_modifyAttrsFile(papi_attribute_t **attrs, char *file)
2420Sstevel@tonic-gate 
2430Sstevel@tonic-gate {
2440Sstevel@tonic-gate 	papi_status_t result = PAPI_OK;
2450Sstevel@tonic-gate 	papi_attribute_t **newAttrs = NULL;
2460Sstevel@tonic-gate 	struct stat   tmpBuf;
2470Sstevel@tonic-gate 	FILE *fd = NULL;
2480Sstevel@tonic-gate 
2492264Sjacobs 	if ((attrs != NULL) && (*attrs != NULL) && (file != NULL)) {
2500Sstevel@tonic-gate 
2510Sstevel@tonic-gate 		/*
2520Sstevel@tonic-gate 		 * check file exist before try to modify it, if it doesn't
2530Sstevel@tonic-gate 		 * exist assume there is an error
2540Sstevel@tonic-gate 		 */
2552264Sjacobs 		if (stat(file, &tmpBuf) == 0) {
2560Sstevel@tonic-gate 			/*
2570Sstevel@tonic-gate 			 * if file is currently empty just write the given
2580Sstevel@tonic-gate 			 * attributes to the file otherwise exact the attributes
2590Sstevel@tonic-gate 			 * from the file and modify them accordingly before
2600Sstevel@tonic-gate 			 * writing them back to the file
2610Sstevel@tonic-gate 			 */
2622264Sjacobs 			if (tmpBuf.st_size == 0) {
2630Sstevel@tonic-gate 				newAttrs = (papi_attribute_t **)attrs;
2640Sstevel@tonic-gate 
2650Sstevel@tonic-gate 				fd = fopen(file, "w");
2662264Sjacobs 				if (fd != NULL) {
2670Sstevel@tonic-gate 					papiAttributeListPrint(fd,
2682264Sjacobs 							newAttrs, "");
2690Sstevel@tonic-gate 					fclose(fd);
2702264Sjacobs 				} else {
2710Sstevel@tonic-gate 					result = PAPI_NOT_POSSIBLE;
2720Sstevel@tonic-gate 				}
2732264Sjacobs 			} else {
2740Sstevel@tonic-gate 				result =
2750Sstevel@tonic-gate 				    psm_modifyAttrsList(file, attrs, &newAttrs);
2760Sstevel@tonic-gate 
2770Sstevel@tonic-gate 				fd = fopen(file, "w");
2782264Sjacobs 				if (fd != NULL) {
2790Sstevel@tonic-gate 					papiAttributeListPrint(fd,
2802264Sjacobs 								newAttrs, "");
2810Sstevel@tonic-gate 					fclose(fd);
2822264Sjacobs 				} else {
2830Sstevel@tonic-gate 					result = PAPI_NOT_POSSIBLE;
2840Sstevel@tonic-gate 				}
2850Sstevel@tonic-gate 
2860Sstevel@tonic-gate 				papiAttributeListFree(newAttrs);
2870Sstevel@tonic-gate 			}
2882264Sjacobs 		} else {
2890Sstevel@tonic-gate 			result = PAPI_NOT_POSSIBLE;
2900Sstevel@tonic-gate 		}
2910Sstevel@tonic-gate 	}
2920Sstevel@tonic-gate 
2930Sstevel@tonic-gate 	return (result);
2940Sstevel@tonic-gate } /* psm_modifyAttrsFile */
2950Sstevel@tonic-gate 
2960Sstevel@tonic-gate 
2970Sstevel@tonic-gate /*
2980Sstevel@tonic-gate  * *****************************************************************************
2990Sstevel@tonic-gate  *
3000Sstevel@tonic-gate  * Description: Extracts the attributes in the given attribute 'file' and
3010Sstevel@tonic-gate  *              creates a new list 'newAttrs' containing the modified list of
3020Sstevel@tonic-gate  *              attributes.
3030Sstevel@tonic-gate  *
3040Sstevel@tonic-gate  * Parameters:  file  - pathname of file containing attributes to be modified
3050Sstevel@tonic-gate  *              attrs - list of attributes and their values to modify
3060Sstevel@tonic-gate  *              newAttrs - returns the modified list of attributes
3070Sstevel@tonic-gate  *
3080Sstevel@tonic-gate  * *****************************************************************************
3090Sstevel@tonic-gate  */
3100Sstevel@tonic-gate 
3110Sstevel@tonic-gate static papi_status_t
3122264Sjacobs psm_modifyAttrsList(char *file, papi_attribute_t **attrs,
3130Sstevel@tonic-gate     papi_attribute_t ***newAttrs)
3140Sstevel@tonic-gate 
3150Sstevel@tonic-gate {
3160Sstevel@tonic-gate 	papi_status_t result = PAPI_OK;
3170Sstevel@tonic-gate 	papi_attribute_t  *nextAttr = NULL;
3180Sstevel@tonic-gate 	papi_attribute_value_t  **values = NULL;
3190Sstevel@tonic-gate 	void *iter = NULL;
3200Sstevel@tonic-gate 	FILE *fd = NULL;
3210Sstevel@tonic-gate 	register int fD = 0;
3220Sstevel@tonic-gate 	char aBuff[200];
3230Sstevel@tonic-gate 	char *a = NULL;
3240Sstevel@tonic-gate 	char *p = NULL;
3250Sstevel@tonic-gate 	int count = 0;
3260Sstevel@tonic-gate 	int n = 0;
3270Sstevel@tonic-gate 
3280Sstevel@tonic-gate 	fd = fopen(file, "r");
3292264Sjacobs 	if (fd != NULL) {
3300Sstevel@tonic-gate 		fD = fileno(fd);
3310Sstevel@tonic-gate 		a = &aBuff[0];
3320Sstevel@tonic-gate 		p = &aBuff[0];
3330Sstevel@tonic-gate 		count = read(fD, &aBuff[0], sizeof (aBuff) - 1);
3342264Sjacobs 		while ((result == PAPI_OK) && (count > 0)) {
3350Sstevel@tonic-gate 			aBuff[count+n] = '\0';
3362264Sjacobs 			if (count == sizeof (aBuff) - n - 1) {
3370Sstevel@tonic-gate 				p = strrchr(aBuff, '\n');
3382264Sjacobs 				if (p != NULL) {
3390Sstevel@tonic-gate 					/* terminate at last complete line */
3400Sstevel@tonic-gate 					*p = '\0';
3410Sstevel@tonic-gate 				}
3420Sstevel@tonic-gate 			}
3430Sstevel@tonic-gate 			result = papiAttributeListFromString(
3440Sstevel@tonic-gate 				newAttrs, PAPI_ATTR_EXCL, aBuff);
3450Sstevel@tonic-gate 
3462264Sjacobs 			if (result == PAPI_OK) {
3470Sstevel@tonic-gate 				/*
3480Sstevel@tonic-gate 				 * handle any part lines and then read the next
3490Sstevel@tonic-gate 				 * buffer from the file
3500Sstevel@tonic-gate 				 */
3510Sstevel@tonic-gate 				n = 0;
3522264Sjacobs 				if (p != a) {
3530Sstevel@tonic-gate 					p++; /* skip NL */
3540Sstevel@tonic-gate 					n = sizeof (aBuff) - 1 - (p - a);
3550Sstevel@tonic-gate 					strncpy(aBuff, p, n);
3560Sstevel@tonic-gate 				}
3570Sstevel@tonic-gate 				count = read(fD, &aBuff[n],
3580Sstevel@tonic-gate 					sizeof (aBuff) - n - 1);
3590Sstevel@tonic-gate 				p = &aBuff[0];
3600Sstevel@tonic-gate 			}
3610Sstevel@tonic-gate 		}
3620Sstevel@tonic-gate 		fclose(fd);
3630Sstevel@tonic-gate 	}
3640Sstevel@tonic-gate 
3650Sstevel@tonic-gate 	/* now modify the attribute list with the new attributes in 'attrs' */
3660Sstevel@tonic-gate 
3670Sstevel@tonic-gate 	nextAttr = papiAttributeListGetNext((papi_attribute_t **)attrs, &iter);
3682264Sjacobs 	while ((result == PAPI_OK) && (nextAttr != NULL)) {
3690Sstevel@tonic-gate 		values = nextAttr->values;
3700Sstevel@tonic-gate 
3712264Sjacobs 		if ((values != NULL) && (*values != NULL)) {
3722264Sjacobs 			result = papiAttributeListAddValue(newAttrs,
3730Sstevel@tonic-gate 						    PAPI_ATTR_REPLACE,
3740Sstevel@tonic-gate 						    nextAttr->name,
3750Sstevel@tonic-gate 						    nextAttr->type, *values);
3760Sstevel@tonic-gate 			values++;
3770Sstevel@tonic-gate 		}
3780Sstevel@tonic-gate 
3790Sstevel@tonic-gate 		while ((result == PAPI_OK) &&
3802264Sjacobs 			(values != NULL) && (*values != NULL)) {
3812264Sjacobs 			result = papiAttributeListAddValue(newAttrs,
3820Sstevel@tonic-gate 						    PAPI_ATTR_APPEND,
3830Sstevel@tonic-gate 						    nextAttr->name,
3840Sstevel@tonic-gate 						    nextAttr->type, *values);
3850Sstevel@tonic-gate 			values++;
3860Sstevel@tonic-gate 		}
3870Sstevel@tonic-gate 		nextAttr =
3880Sstevel@tonic-gate 		    papiAttributeListGetNext((papi_attribute_t **)attrs, &iter);
3890Sstevel@tonic-gate 	}
3900Sstevel@tonic-gate 
3910Sstevel@tonic-gate 	return (result);
3920Sstevel@tonic-gate } /* papi_modifyAttrsList() */
3930Sstevel@tonic-gate #endif
3940Sstevel@tonic-gate 
3950Sstevel@tonic-gate 
3960Sstevel@tonic-gate papi_status_t
3972264Sjacobs papiJobSubmit(papi_service_t handle, char *printer,
3982264Sjacobs 		papi_attribute_t **job_attributes,
3992264Sjacobs 		papi_job_ticket_t *job_ticket,
4002264Sjacobs 		char **files, papi_job_t *job)
4010Sstevel@tonic-gate {
4020Sstevel@tonic-gate 	papi_status_t status;
4030Sstevel@tonic-gate 	service_t *svc = handle;
4040Sstevel@tonic-gate 	job_t *j;
4050Sstevel@tonic-gate 	int file_no;
4060Sstevel@tonic-gate 	char *request_id = NULL;
4070Sstevel@tonic-gate 	REQUEST *request;
4080Sstevel@tonic-gate 	int i;
4090Sstevel@tonic-gate 	char *c;
4100Sstevel@tonic-gate 	char *tmp = NULL;
4110Sstevel@tonic-gate 	char lpfile[BUFSIZ];
4120Sstevel@tonic-gate 
4130Sstevel@tonic-gate 	if ((svc == NULL) || (printer == NULL) || (files == NULL) ||
4140Sstevel@tonic-gate 	    (job == NULL))
4150Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
4160Sstevel@tonic-gate 
4170Sstevel@tonic-gate 	if (job_ticket != NULL)
4180Sstevel@tonic-gate 		return (PAPI_OPERATION_NOT_SUPPORTED);
4190Sstevel@tonic-gate 
4200Sstevel@tonic-gate 	if (files != NULL)
4210Sstevel@tonic-gate 		for (file_no = 0; files[file_no] != NULL; file_no++)
4220Sstevel@tonic-gate 			if (access(files[file_no], R_OK) < 0) {
4230Sstevel@tonic-gate 				detailed_error(svc,
4240Sstevel@tonic-gate 					gettext("Cannot access file: %s: %s"),
4250Sstevel@tonic-gate 					files[file_no], strerror(errno));
4260Sstevel@tonic-gate 				return (PAPI_BAD_ARGUMENT);
4270Sstevel@tonic-gate 			}
4280Sstevel@tonic-gate 
4290Sstevel@tonic-gate 	if ((*job = j = calloc(1, sizeof (*j))) == NULL)
4300Sstevel@tonic-gate 		return (PAPI_TEMPORARY_ERROR);
4310Sstevel@tonic-gate 
4320Sstevel@tonic-gate 	/* file_no + 1 for the control file (-0) */
4330Sstevel@tonic-gate 	status = lpsched_alloc_files(svc, file_no + 1, &request_id);
4340Sstevel@tonic-gate 	if (status != PAPI_OK)
4350Sstevel@tonic-gate 		return (status);
4360Sstevel@tonic-gate 
4370Sstevel@tonic-gate 	request = create_request(svc, (char *)printer,
4380Sstevel@tonic-gate 				(papi_attribute_t **)job_attributes);
4390Sstevel@tonic-gate 
4400Sstevel@tonic-gate 	for (i = 0; files[i] != NULL; i++) {
4410Sstevel@tonic-gate 		papi_status_t status;
4420Sstevel@tonic-gate 		snprintf(lpfile, sizeof (lpfile), "%s%s-%d",
4430Sstevel@tonic-gate 			"/var/spool/lp/temp/", request_id, i+1);
4440Sstevel@tonic-gate 		status = copy_file(files[i], lpfile);
4450Sstevel@tonic-gate 		if (status != PAPI_OK) {
4460Sstevel@tonic-gate 			detailed_error(svc,
4470Sstevel@tonic-gate 				gettext("unable to copy: %s -> %s: %s"),
4480Sstevel@tonic-gate 				files[i], lpfile, strerror(errno));
4490Sstevel@tonic-gate 				freerequest(request);
4500Sstevel@tonic-gate 			return (PAPI_DEVICE_ERROR);
4510Sstevel@tonic-gate 		}
4520Sstevel@tonic-gate 		addlist(&(request->file_list), lpfile);
4530Sstevel@tonic-gate 	}
4540Sstevel@tonic-gate 
4550Sstevel@tonic-gate #ifdef LP_USE_PAPI_ATTR
4560Sstevel@tonic-gate 	/*
4570Sstevel@tonic-gate 	 * store the job attributes in the PAPI job attribute file that was
4580Sstevel@tonic-gate 	 * created by lpsched_alloc_files(), the attributes will then pass
4590Sstevel@tonic-gate 	 * through lpsched and be given to the slow-filters and the printer's
4600Sstevel@tonic-gate 	 * interface script to process them
4610Sstevel@tonic-gate 	 */
4620Sstevel@tonic-gate 	snprintf(lpfile, sizeof (lpfile), "%s%s-%s",
4630Sstevel@tonic-gate 		"/var/spool/lp/temp/", request_id, LP_PAPIATTRNAME);
4640Sstevel@tonic-gate 	status = psm_copy_attrsToFile(job_attributes, lpfile);
4650Sstevel@tonic-gate 	if (status != PAPI_OK) {
4660Sstevel@tonic-gate 		detailed_error(svc, "unable to copy attributes to file: %s: %s",
4670Sstevel@tonic-gate 				lpfile, strerror(errno));
4680Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
4690Sstevel@tonic-gate 	}
4700Sstevel@tonic-gate #endif
4710Sstevel@tonic-gate 
4720Sstevel@tonic-gate 	/* store the meta-data file */
4730Sstevel@tonic-gate 	snprintf(lpfile, sizeof (lpfile), "%s-0", request_id);
4740Sstevel@tonic-gate 	if (putrequest(lpfile, request) < 0) {
4750Sstevel@tonic-gate 		detailed_error(svc, gettext("unable to save request: %s: %s"),
4760Sstevel@tonic-gate 			lpfile, strerror(errno));
4770Sstevel@tonic-gate 		freerequest(request);
4780Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
4790Sstevel@tonic-gate 	}
4800Sstevel@tonic-gate 
4810Sstevel@tonic-gate 	status = lpsched_commit_job(svc, lpfile, &tmp);
4820Sstevel@tonic-gate 	if (status != PAPI_OK) {
4830Sstevel@tonic-gate 		unlink(lpfile);
4840Sstevel@tonic-gate 		freerequest(request);
4850Sstevel@tonic-gate 		return (status);
4860Sstevel@tonic-gate 	}
4870Sstevel@tonic-gate 
4880Sstevel@tonic-gate 	lpsched_request_to_job_attributes(request, j);
4890Sstevel@tonic-gate 	freerequest(request);
4900Sstevel@tonic-gate 
4910Sstevel@tonic-gate 	if ((c = strrchr(tmp, '-')) != NULL)
4920Sstevel@tonic-gate 		c++;
4930Sstevel@tonic-gate 	papiAttributeListAddInteger(&j->attributes, PAPI_ATTR_REPLACE,
4940Sstevel@tonic-gate 			"job-id", atoi(c));
4950Sstevel@tonic-gate 	papiAttributeListAddString(&j->attributes, PAPI_ATTR_REPLACE,
4960Sstevel@tonic-gate 			"job-uri", tmp);
4970Sstevel@tonic-gate 
4980Sstevel@tonic-gate 	return (PAPI_OK);
4990Sstevel@tonic-gate }
5000Sstevel@tonic-gate 
5010Sstevel@tonic-gate papi_status_t
5022264Sjacobs papiJobSubmitByReference(papi_service_t handle, char *printer,
5032264Sjacobs 		papi_attribute_t **job_attributes,
5042264Sjacobs 		papi_job_ticket_t *job_ticket,
5052264Sjacobs 		char **files, papi_job_t *job)
5060Sstevel@tonic-gate {
5070Sstevel@tonic-gate 	service_t *svc = handle;
5080Sstevel@tonic-gate 	job_t *j;
5090Sstevel@tonic-gate 	int file_no;
5100Sstevel@tonic-gate 	short status;
5110Sstevel@tonic-gate 	char *request_id = NULL;
5120Sstevel@tonic-gate 	REQUEST *request;
5130Sstevel@tonic-gate 	char *c;
5140Sstevel@tonic-gate 	char *tmp = NULL;
5150Sstevel@tonic-gate 	char lpfile[BUFSIZ];
5160Sstevel@tonic-gate 	char **file_list = NULL;
5170Sstevel@tonic-gate 
5180Sstevel@tonic-gate 	if ((svc == NULL) || (printer == NULL) || (files == NULL) ||
5190Sstevel@tonic-gate 	    (job == NULL))
5200Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
5210Sstevel@tonic-gate 
5220Sstevel@tonic-gate 	if (job_ticket != NULL)
5230Sstevel@tonic-gate 		return (PAPI_OPERATION_NOT_SUPPORTED);
5240Sstevel@tonic-gate 
5250Sstevel@tonic-gate 	if (files != NULL)
5260Sstevel@tonic-gate 		for (file_no = 0; files[file_no] != NULL; file_no++) {
5270Sstevel@tonic-gate 			if (access(files[file_no], R_OK) < 0) {
5280Sstevel@tonic-gate 				detailed_error(svc,
5290Sstevel@tonic-gate 					gettext("Cannot access file: %s: %s"),
5300Sstevel@tonic-gate 					files[file_no], strerror(errno));
531*2660Sjacobs 				return (PAPI_DOCUMENT_ACCESS_ERROR);
5320Sstevel@tonic-gate 			}
533*2660Sjacobs 			if (files[file_no][0] != '/') {
534*2660Sjacobs 				char path[MAXPATHLEN];
535*2660Sjacobs 
536*2660Sjacobs 				if (getcwd(path, sizeof (path)) == NULL) {
537*2660Sjacobs 					detailed_error(svc, gettext(
538*2660Sjacobs 						"getcwd for file: %s: %s"),
539*2660Sjacobs 						files[file_no],
540*2660Sjacobs 						strerror(errno));
541*2660Sjacobs 					return (PAPI_DOCUMENT_ACCESS_ERROR);
542*2660Sjacobs 				}
543*2660Sjacobs 				strlcat(path, "/", sizeof (path));
544*2660Sjacobs 				if (strlcat(path, files[file_no], sizeof (path))
545*2660Sjacobs 						>= sizeof (path)) {
546*2660Sjacobs 					detailed_error(svc, gettext(
547*2660Sjacobs 						"pathname too long: %s"),
548*2660Sjacobs 						files[file_no]);
549*2660Sjacobs 					return (PAPI_DOCUMENT_ACCESS_ERROR);
550*2660Sjacobs 				}
551*2660Sjacobs 				addlist(&file_list, path);
552*2660Sjacobs 			} else
553*2660Sjacobs 				addlist(&file_list, (char *)files[file_no]);
5540Sstevel@tonic-gate 		}
5550Sstevel@tonic-gate 
5560Sstevel@tonic-gate 	if ((*job = j = calloc(1, sizeof (*j))) == NULL)
5570Sstevel@tonic-gate 		return (PAPI_TEMPORARY_ERROR);
5580Sstevel@tonic-gate 
5590Sstevel@tonic-gate 	/* 1 for the control file (-0) */
5600Sstevel@tonic-gate 	status = lpsched_alloc_files(svc, 1, &request_id);
5610Sstevel@tonic-gate 	if (status != PAPI_OK)
5620Sstevel@tonic-gate 		return (status);
5630Sstevel@tonic-gate 
5640Sstevel@tonic-gate 	request = create_request(svc, (char *)printer,
5650Sstevel@tonic-gate 				(papi_attribute_t **)job_attributes);
5660Sstevel@tonic-gate 	request->file_list = file_list;
5670Sstevel@tonic-gate 
5680Sstevel@tonic-gate #ifdef LP_USE_PAPI_ATTR
5690Sstevel@tonic-gate 	/*
5700Sstevel@tonic-gate 	 * store the job attributes in the PAPI job attribute file that was
5710Sstevel@tonic-gate 	 * created by lpsched_alloc_files(), the attributes will then pass
5720Sstevel@tonic-gate 	 * through lpsched and be given to the slow-filters and the printer's
5730Sstevel@tonic-gate 	 * interface script to process them
5740Sstevel@tonic-gate 	 */
5750Sstevel@tonic-gate 	snprintf(lpfile, sizeof (lpfile), "%s%s-%s",
5760Sstevel@tonic-gate 		"/var/spool/lp/temp/", request_id, LP_PAPIATTRNAME);
5770Sstevel@tonic-gate 	status = psm_copy_attrsToFile(job_attributes, lpfile);
5780Sstevel@tonic-gate 	if (status != PAPI_OK) {
5790Sstevel@tonic-gate 		detailed_error(svc, "unable to copy attributes to file: %s: %s",
5800Sstevel@tonic-gate 				lpfile, strerror(errno));
5810Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
5820Sstevel@tonic-gate 	}
5830Sstevel@tonic-gate #endif
5840Sstevel@tonic-gate 
5850Sstevel@tonic-gate 	/* store the meta-data file */
5860Sstevel@tonic-gate 	snprintf(lpfile, sizeof (lpfile), "%s-0", request_id);
5870Sstevel@tonic-gate 	if (putrequest(lpfile, request) < 0) {
5880Sstevel@tonic-gate 		detailed_error(svc, gettext("unable to save request: %s: %s"),
5890Sstevel@tonic-gate 			lpfile, strerror(errno));
5900Sstevel@tonic-gate 		freerequest(request);
5910Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
5920Sstevel@tonic-gate 	}
5930Sstevel@tonic-gate 
5940Sstevel@tonic-gate 	status = lpsched_commit_job(svc, lpfile, &tmp);
5950Sstevel@tonic-gate 	if (status != PAPI_OK) {
5960Sstevel@tonic-gate 		unlink(lpfile);
5970Sstevel@tonic-gate 		freerequest(request);
5980Sstevel@tonic-gate 		return (status);
5990Sstevel@tonic-gate 	}
6000Sstevel@tonic-gate 
6010Sstevel@tonic-gate 	lpsched_request_to_job_attributes(request, j);
6020Sstevel@tonic-gate 
6030Sstevel@tonic-gate 	freerequest(request);
6040Sstevel@tonic-gate 
6050Sstevel@tonic-gate 	if ((c = strrchr(tmp, '-')) != NULL)
6060Sstevel@tonic-gate 		c++;
6070Sstevel@tonic-gate 	papiAttributeListAddInteger(&j->attributes, PAPI_ATTR_REPLACE,
6080Sstevel@tonic-gate 			"job-id", atoi(c));
6090Sstevel@tonic-gate 	papiAttributeListAddString(&j->attributes, PAPI_ATTR_REPLACE,
6100Sstevel@tonic-gate 			"job-uri", tmp);
6110Sstevel@tonic-gate 
6120Sstevel@tonic-gate 	return (PAPI_OK);
6130Sstevel@tonic-gate }
6140Sstevel@tonic-gate 
6150Sstevel@tonic-gate papi_status_t
6162264Sjacobs papiJobValidate(papi_service_t handle, char *printer,
6172264Sjacobs 		papi_attribute_t **job_attributes,
6182264Sjacobs 		papi_job_ticket_t *job_ticket,
6192264Sjacobs 		char **files, papi_job_t *job)
6200Sstevel@tonic-gate {
6210Sstevel@tonic-gate 	papi_status_t status;
6220Sstevel@tonic-gate 	papi_attribute_t **attributes = NULL;
6230Sstevel@tonic-gate 	int i;
6240Sstevel@tonic-gate 
6250Sstevel@tonic-gate 	papiAttributeListAddString(&attributes, PAPI_ATTR_REPLACE,
6260Sstevel@tonic-gate 			"job-hold-until", "hold");
6270Sstevel@tonic-gate 	for (i = 0; job_attributes[i]; i++)
6280Sstevel@tonic-gate 		list_append(&attributes, job_attributes[i]);
6290Sstevel@tonic-gate 
6300Sstevel@tonic-gate 	status = papiJobSubmitByReference(handle, printer,
6312264Sjacobs 				(papi_attribute_t **)attributes,
6320Sstevel@tonic-gate 				job_ticket, files, job);
6330Sstevel@tonic-gate 	if (status == PAPI_OK) {
6340Sstevel@tonic-gate 		int id = papiJobGetId(*job);
6350Sstevel@tonic-gate 
6360Sstevel@tonic-gate 		if (id != -1)
6370Sstevel@tonic-gate 			papiJobCancel(handle, printer, id);
6380Sstevel@tonic-gate 	}
6390Sstevel@tonic-gate 
6400Sstevel@tonic-gate 	attributes[1] = NULL;	/* after attr[0], they are in another list */
6410Sstevel@tonic-gate 	papiAttributeListFree(attributes);
6420Sstevel@tonic-gate 
6430Sstevel@tonic-gate 	return (status);
6440Sstevel@tonic-gate }
6450Sstevel@tonic-gate 
6460Sstevel@tonic-gate papi_status_t
6472264Sjacobs papiJobStreamOpen(papi_service_t handle, char *printer,
6482264Sjacobs 		papi_attribute_t **job_attributes,
6492264Sjacobs 		papi_job_ticket_t *job_ticket, papi_stream_t *stream)
6500Sstevel@tonic-gate {
6510Sstevel@tonic-gate 	papi_status_t status;
6520Sstevel@tonic-gate 	service_t *svc = handle;
6530Sstevel@tonic-gate 	job_stream_t *s = NULL;
6540Sstevel@tonic-gate 	char *request_id = NULL;
6550Sstevel@tonic-gate 	char lpfile[BUFSIZ];
6560Sstevel@tonic-gate 
6570Sstevel@tonic-gate 	if ((svc == NULL) || (printer == NULL) || (stream == NULL))
6580Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
6590Sstevel@tonic-gate 
6600Sstevel@tonic-gate 	if (job_ticket != NULL)
6610Sstevel@tonic-gate 		return (PAPI_OPERATION_NOT_SUPPORTED);
6620Sstevel@tonic-gate 
6630Sstevel@tonic-gate 	if ((*stream = s = calloc(1, sizeof (*s))) == NULL)
6640Sstevel@tonic-gate 		return (PAPI_TEMPORARY_ERROR);
6650Sstevel@tonic-gate 
6660Sstevel@tonic-gate 	/* 1 for data, 1 for the meta-data (-0) */
6670Sstevel@tonic-gate 	status = lpsched_alloc_files(svc, 2, &request_id);
6680Sstevel@tonic-gate 	if (status != PAPI_OK)
6690Sstevel@tonic-gate 		return (status);
6700Sstevel@tonic-gate 
6710Sstevel@tonic-gate 	s->request = create_request(svc, (char *)printer,
6720Sstevel@tonic-gate 				(papi_attribute_t **)job_attributes);
6730Sstevel@tonic-gate 	snprintf(lpfile, sizeof (lpfile), "/var/spool/lp/temp/%s-1",
6740Sstevel@tonic-gate 							request_id);
6750Sstevel@tonic-gate 	s->fd = open(lpfile, O_WRONLY);
6760Sstevel@tonic-gate 	addlist(&(s->request->file_list), lpfile);
6770Sstevel@tonic-gate 
6780Sstevel@tonic-gate #ifdef LP_USE_PAPI_ATTR
6790Sstevel@tonic-gate 	/*
6800Sstevel@tonic-gate 	 * store the job attributes in the PAPI job attribute file that was
6810Sstevel@tonic-gate 	 * created by lpsched_alloc_files(), the attributes will then pass
6820Sstevel@tonic-gate 	 * through lpsched and be given to the slow-filters and the printer's
6830Sstevel@tonic-gate 	 * interface script to process them
6840Sstevel@tonic-gate 	 */
6850Sstevel@tonic-gate 	snprintf(lpfile, sizeof (lpfile), "%s%s-%s",
6860Sstevel@tonic-gate 		"/var/spool/lp/temp/", request_id, LP_PAPIATTRNAME);
6870Sstevel@tonic-gate 	status = psm_copy_attrsToFile(job_attributes, lpfile);
6880Sstevel@tonic-gate 	if (status != PAPI_OK) {
6890Sstevel@tonic-gate 		detailed_error(svc, "unable to copy attributes to file: %s: %s",
6900Sstevel@tonic-gate 				lpfile, strerror(errno));
6910Sstevel@tonic-gate 		close(s->fd);
6920Sstevel@tonic-gate 		free(s);
6930Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
6940Sstevel@tonic-gate 	}
6950Sstevel@tonic-gate #endif
6960Sstevel@tonic-gate 
6970Sstevel@tonic-gate 	/* store the meta-data file */
6980Sstevel@tonic-gate 	snprintf(lpfile, sizeof (lpfile), "%s-0", request_id);
6990Sstevel@tonic-gate 	s->meta_data_file = strdup(lpfile);
7000Sstevel@tonic-gate 	if (putrequest(lpfile, s->request) < 0) {
7010Sstevel@tonic-gate 		detailed_error(svc, gettext("unable to save request: %s: %s"),
7020Sstevel@tonic-gate 			lpfile, strerror(errno));
7030Sstevel@tonic-gate 		s->request = NULL;
7040Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
7050Sstevel@tonic-gate 	}
7060Sstevel@tonic-gate 
7070Sstevel@tonic-gate 	return (PAPI_OK);
7080Sstevel@tonic-gate }
7090Sstevel@tonic-gate 
7100Sstevel@tonic-gate papi_status_t
7110Sstevel@tonic-gate papiJobStreamWrite(papi_service_t handle,
7122264Sjacobs 		papi_stream_t stream, void *buffer, size_t buflen)
7130Sstevel@tonic-gate {
7140Sstevel@tonic-gate 	service_t *svc = handle;
7150Sstevel@tonic-gate 	job_stream_t *s = stream;
7160Sstevel@tonic-gate 
7170Sstevel@tonic-gate 	if ((svc == NULL) || (stream == NULL) || (buffer == NULL))
7180Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
7190Sstevel@tonic-gate 
7200Sstevel@tonic-gate 	if (write(s->fd, buffer, buflen) != buflen)
7210Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
7220Sstevel@tonic-gate 
7230Sstevel@tonic-gate 	return (PAPI_OK);
7240Sstevel@tonic-gate }
7250Sstevel@tonic-gate papi_status_t
7260Sstevel@tonic-gate papiJobStreamClose(papi_service_t handle,
7270Sstevel@tonic-gate 		papi_stream_t stream, papi_job_t *job)
7280Sstevel@tonic-gate {
7290Sstevel@tonic-gate 	papi_status_t status = PAPI_OK;
7300Sstevel@tonic-gate 	service_t *svc = handle;
7310Sstevel@tonic-gate 	job_stream_t *s = stream;
7320Sstevel@tonic-gate 	job_t *j = NULL;
7330Sstevel@tonic-gate 	char *tmp = NULL, *c;
7340Sstevel@tonic-gate 
7350Sstevel@tonic-gate 	if ((svc == NULL) || (stream == NULL) || (job == NULL))
7360Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
7370Sstevel@tonic-gate 
7380Sstevel@tonic-gate 	if ((*job = j = calloc(1, sizeof (*j))) == NULL)
7390Sstevel@tonic-gate 		return (PAPI_TEMPORARY_ERROR);
7400Sstevel@tonic-gate 
7410Sstevel@tonic-gate 	close(s->fd);
7420Sstevel@tonic-gate 
7430Sstevel@tonic-gate 	lpsched_request_to_job_attributes(s->request, j);
7440Sstevel@tonic-gate 
7450Sstevel@tonic-gate 	if (s->meta_data_file != NULL) {
7460Sstevel@tonic-gate 		status = lpsched_commit_job(svc, s->meta_data_file, &tmp);
7470Sstevel@tonic-gate 		if (status != PAPI_OK) {
7480Sstevel@tonic-gate 			unlink(s->meta_data_file);
7490Sstevel@tonic-gate 			return (status);
7500Sstevel@tonic-gate 		}
7510Sstevel@tonic-gate 		if ((c = strrchr(tmp, '-')) != NULL)
7520Sstevel@tonic-gate 			c++;
7530Sstevel@tonic-gate 		papiAttributeListAddInteger(&j->attributes, PAPI_ATTR_REPLACE,
7540Sstevel@tonic-gate 			"job-id", atoi(c));
7550Sstevel@tonic-gate 		papiAttributeListAddString(&j->attributes, PAPI_ATTR_REPLACE,
7560Sstevel@tonic-gate 			"job-uri", tmp);
7570Sstevel@tonic-gate 		free(s->meta_data_file);
7580Sstevel@tonic-gate 	}
7590Sstevel@tonic-gate 	free(s);
7600Sstevel@tonic-gate 
7610Sstevel@tonic-gate 	return (PAPI_OK);
7620Sstevel@tonic-gate }
7630Sstevel@tonic-gate 
7640Sstevel@tonic-gate papi_status_t
7652264Sjacobs papiJobQuery(papi_service_t handle, char *printer, int32_t job_id,
7662264Sjacobs 		char **requested_attrs,
7670Sstevel@tonic-gate 		papi_job_t *job)
7680Sstevel@tonic-gate {
7690Sstevel@tonic-gate 	service_t *svc = handle;
7700Sstevel@tonic-gate 	job_t *j;
7710Sstevel@tonic-gate 	char *dest;
7720Sstevel@tonic-gate 	char req_id[32];
7730Sstevel@tonic-gate 	short rc;
7740Sstevel@tonic-gate 	char *form = NULL,
7750Sstevel@tonic-gate 		*request_id = NULL,
7760Sstevel@tonic-gate 		*charset = NULL,
7770Sstevel@tonic-gate 		*user = NULL,
7781676Sjpk 		*slabel = NULL,
7790Sstevel@tonic-gate 		*file = NULL;
7800Sstevel@tonic-gate 	time_t date = 0;
7810Sstevel@tonic-gate 	size_t size = 0;
7820Sstevel@tonic-gate 	short  rank = 0,
7830Sstevel@tonic-gate 		state = 0;
7840Sstevel@tonic-gate 
7850Sstevel@tonic-gate 	if ((handle == NULL) || (printer == NULL) || (job_id < 0))
7860Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
7870Sstevel@tonic-gate 
7880Sstevel@tonic-gate 	dest = printer_name_from_uri_id(printer, job_id);
7890Sstevel@tonic-gate 	snprintf(req_id, sizeof (req_id), "%s-%d", dest, job_id);
7900Sstevel@tonic-gate 	free(dest);
7910Sstevel@tonic-gate 
7920Sstevel@tonic-gate 	rc = snd_msg(svc, S_INQUIRE_REQUEST_RANK, 0, "", "", req_id, "", "");
7930Sstevel@tonic-gate 	if (rc < 0)
7940Sstevel@tonic-gate 		return (PAPI_SERVICE_UNAVAILABLE);
7950Sstevel@tonic-gate 
7960Sstevel@tonic-gate 	if (rcv_msg(svc, R_INQUIRE_REQUEST_RANK, &rc, &request_id,
7971676Sjpk 			&user, &slabel, &size, &date, &state, &dest, &form,
7980Sstevel@tonic-gate 			&charset, &rank, &file) < 0) {
7990Sstevel@tonic-gate 		detailed_error(svc,
8000Sstevel@tonic-gate 			gettext("failed to read response from scheduler"));
8010Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
8020Sstevel@tonic-gate 	}
8030Sstevel@tonic-gate 
8040Sstevel@tonic-gate 	if ((request_id == NULL) || (request_id[0] == NULL))
8050Sstevel@tonic-gate 		return (PAPI_NOT_FOUND);
8060Sstevel@tonic-gate 
8070Sstevel@tonic-gate 	if ((*job = j = calloc(1, sizeof (*j))) == NULL)
8080Sstevel@tonic-gate 		return (PAPI_TEMPORARY_ERROR);
8090Sstevel@tonic-gate 
8101676Sjpk 	job_status_to_attributes(j, request_id, user, slabel, size, date, state,
8110Sstevel@tonic-gate 				dest, form, charset, rank, file);
8120Sstevel@tonic-gate 
8130Sstevel@tonic-gate 	snprintf(req_id, sizeof (req_id), "%d-0", job_id);
8140Sstevel@tonic-gate 	lpsched_read_job_configuration(svc, j, req_id);
8150Sstevel@tonic-gate 
8160Sstevel@tonic-gate 	return (PAPI_OK);
8170Sstevel@tonic-gate }
8180Sstevel@tonic-gate 
8190Sstevel@tonic-gate papi_status_t
8202264Sjacobs papiJobMove(papi_service_t handle, char *printer, int32_t job_id,
8212264Sjacobs 		char *destination)
8222264Sjacobs {
8232264Sjacobs 	papi_status_t result = PAPI_OK;
8242264Sjacobs 	service_t *svc = handle;
8252264Sjacobs 	char req_id[64];
8262264Sjacobs 	char *queue;
8272264Sjacobs 	char *user = NULL;
8282264Sjacobs 
8292264Sjacobs 	if ((svc == NULL) || (printer == NULL) || (job_id < 0) ||
8302264Sjacobs 	    (destination == NULL))
8312264Sjacobs 		return (PAPI_BAD_ARGUMENT);
8322264Sjacobs 
8332264Sjacobs 	queue = printer_name_from_uri_id(printer, job_id);
8342264Sjacobs 	snprintf(req_id, sizeof (req_id), "%s-%d", queue, job_id);
8352264Sjacobs 	free(queue);
8362264Sjacobs 
8372264Sjacobs 	if (papiAttributeListGetString(svc->attributes, NULL, "user-name",
8382264Sjacobs 			&user) == PAPI_OK) {
8392264Sjacobs 		REQUEST *r = getrequest(req_id);
8402264Sjacobs 
8412264Sjacobs 		if ((r != NULL) && (r->user != NULL) &&
8422264Sjacobs 		    (strcmp(r->user, user) != 0))
8432264Sjacobs 			result = PAPI_NOT_AUTHORIZED;
8442264Sjacobs 		freerequest(r);
8452264Sjacobs 	}
8462264Sjacobs 
8472264Sjacobs 	if (result == PAPI_OK) {
8482264Sjacobs 		short status = MOK;
8492264Sjacobs 		char *dest = printer_name_from_uri_id(destination, -1);
8502264Sjacobs 
8512264Sjacobs 		if ((snd_msg(svc, S_MOVE_REQUEST, req_id, dest) < 0) ||
8522264Sjacobs 		    (rcv_msg(svc, R_MOVE_REQUEST, &status) < 0))
8532264Sjacobs 			status = MTRANSMITERR;
8542264Sjacobs 
8552264Sjacobs 		free(dest);
8562264Sjacobs 
8572264Sjacobs 		result = lpsched_status_to_papi_status(status);
8582264Sjacobs 	}
8592264Sjacobs 
8602264Sjacobs 	return (result);
8612264Sjacobs }
8622264Sjacobs 
8632264Sjacobs papi_status_t
8642264Sjacobs papiJobCancel(papi_service_t handle, char *printer, int32_t job_id)
8650Sstevel@tonic-gate {
8660Sstevel@tonic-gate 	papi_status_t result = PAPI_OK;
8670Sstevel@tonic-gate 	service_t *svc = handle;
8680Sstevel@tonic-gate 	char req_id[64];
8690Sstevel@tonic-gate 	char *dest;
8700Sstevel@tonic-gate 	char *user = NULL;
8710Sstevel@tonic-gate 
8720Sstevel@tonic-gate 	if ((svc == NULL) || (printer == NULL) || (job_id < 0))
8730Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
8740Sstevel@tonic-gate 
8750Sstevel@tonic-gate 	dest = printer_name_from_uri_id(printer, job_id);
8760Sstevel@tonic-gate 	snprintf(req_id, sizeof (req_id), "%s-%d", dest, job_id);
8770Sstevel@tonic-gate 	free(dest);
8780Sstevel@tonic-gate 
8790Sstevel@tonic-gate 	if (papiAttributeListGetString(svc->attributes, NULL, "user-name",
8800Sstevel@tonic-gate 			&user) == PAPI_OK) {
8810Sstevel@tonic-gate 		REQUEST *r = getrequest(req_id);
8820Sstevel@tonic-gate 
8832264Sjacobs 		if ((r != NULL) && (r->user != NULL) &&
8842264Sjacobs 		    (strcmp(r->user, user) != 0))
8850Sstevel@tonic-gate 			result = PAPI_NOT_AUTHORIZED;
8860Sstevel@tonic-gate 		freerequest(r);
8870Sstevel@tonic-gate 	}
8880Sstevel@tonic-gate 
8890Sstevel@tonic-gate 	if (result == PAPI_OK) {
8900Sstevel@tonic-gate 		short status = MOK;
8910Sstevel@tonic-gate 
8920Sstevel@tonic-gate 		if ((snd_msg(svc, S_CANCEL_REQUEST, req_id) < 0) ||
8930Sstevel@tonic-gate 		    (rcv_msg(svc, R_CANCEL_REQUEST, &status) < 0))
8940Sstevel@tonic-gate 			status = MTRANSMITERR;
8950Sstevel@tonic-gate 
8960Sstevel@tonic-gate 		result = lpsched_status_to_papi_status(status);
8970Sstevel@tonic-gate 	}
8980Sstevel@tonic-gate 
8990Sstevel@tonic-gate 	return (result);
9000Sstevel@tonic-gate }
9010Sstevel@tonic-gate 
9020Sstevel@tonic-gate papi_status_t
9032264Sjacobs hold_release_job(papi_service_t handle, char *printer,
9042264Sjacobs 		int32_t job_id, int flag)
9050Sstevel@tonic-gate {
9060Sstevel@tonic-gate 	papi_status_t status;
9070Sstevel@tonic-gate 	service_t *svc = handle;
9080Sstevel@tonic-gate 	REQUEST *r = NULL;
9090Sstevel@tonic-gate 	char *file;
9100Sstevel@tonic-gate 	char *dest;
9110Sstevel@tonic-gate 
9120Sstevel@tonic-gate 	if ((svc == NULL) || (printer == NULL) || (job_id < 0))
9130Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
9140Sstevel@tonic-gate 
9150Sstevel@tonic-gate 	if ((status = authorized(svc, job_id)) != PAPI_OK)
9160Sstevel@tonic-gate 		return (status);
9170Sstevel@tonic-gate 
9180Sstevel@tonic-gate 	dest = printer_name_from_uri_id(printer, job_id);
9190Sstevel@tonic-gate 	status = lpsched_start_change(svc, dest, job_id, &file);
9200Sstevel@tonic-gate 	if (status != PAPI_OK)
9210Sstevel@tonic-gate 		return (status);
9220Sstevel@tonic-gate 
9230Sstevel@tonic-gate 	if ((r = getrequest(file)) != NULL) {
9240Sstevel@tonic-gate 		r->actions &= ~ACT_RESUME;
9252264Sjacobs 		switch (flag) {
9262264Sjacobs 		case 0:
9270Sstevel@tonic-gate 			r->actions |= ACT_HOLD;
9282264Sjacobs 			break;
9292264Sjacobs 		case 1:
9300Sstevel@tonic-gate 			r->actions |= ACT_RESUME;
9312264Sjacobs 			break;
9322264Sjacobs 		case 2:
9332264Sjacobs 			r->actions |= ACT_IMMEDIATE;
9342264Sjacobs 			break;
9352264Sjacobs 		}
9360Sstevel@tonic-gate 		if (putrequest(file, r) < 0) {
9370Sstevel@tonic-gate 			detailed_error(svc,
9380Sstevel@tonic-gate 				gettext("failed to write job: %s: %s"),
9390Sstevel@tonic-gate 				file, strerror(errno));
9402264Sjacobs 			freerequest(r);
9410Sstevel@tonic-gate 			return (PAPI_DEVICE_ERROR);
9420Sstevel@tonic-gate 		}
9432264Sjacobs 		freerequest(r);
9440Sstevel@tonic-gate 	} else {
9450Sstevel@tonic-gate 		detailed_error(svc, gettext("failed to read job: %s: %s"),
9460Sstevel@tonic-gate 				file, strerror(errno));
9470Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
9480Sstevel@tonic-gate 	}
9490Sstevel@tonic-gate 
9500Sstevel@tonic-gate 	status = lpsched_end_change(svc, dest, job_id);
9510Sstevel@tonic-gate 
9520Sstevel@tonic-gate 	return (status);
9530Sstevel@tonic-gate }
9540Sstevel@tonic-gate 
9550Sstevel@tonic-gate papi_status_t
9562264Sjacobs papiJobHold(papi_service_t handle, char *printer, int32_t job_id)
9570Sstevel@tonic-gate {
9580Sstevel@tonic-gate 	return (hold_release_job(handle, printer, job_id, 0));
9590Sstevel@tonic-gate }
9600Sstevel@tonic-gate 
9610Sstevel@tonic-gate papi_status_t
9622264Sjacobs papiJobRelease(papi_service_t handle, char *printer, int32_t job_id)
9630Sstevel@tonic-gate {
9640Sstevel@tonic-gate 	return (hold_release_job(handle, printer, job_id, 1));
9650Sstevel@tonic-gate }
9660Sstevel@tonic-gate 
9670Sstevel@tonic-gate papi_status_t
9682264Sjacobs papiJobPromote(papi_service_t handle, char *printer, int32_t job_id)
9690Sstevel@tonic-gate {
9702264Sjacobs 	return (hold_release_job(handle, printer, job_id, 2));
9710Sstevel@tonic-gate }
9720Sstevel@tonic-gate 
9730Sstevel@tonic-gate papi_status_t
9742264Sjacobs papiJobModify(papi_service_t handle, char *printer, int32_t job_id,
9752264Sjacobs 		papi_attribute_t **attributes, papi_job_t *job)
9760Sstevel@tonic-gate {
9770Sstevel@tonic-gate 	papi_status_t status;
9780Sstevel@tonic-gate 	job_t *j = NULL;
9790Sstevel@tonic-gate 	service_t *svc = handle;
9800Sstevel@tonic-gate 	char *file = NULL;
9810Sstevel@tonic-gate 	char *dest;
9820Sstevel@tonic-gate 	REQUEST *r = NULL;
9830Sstevel@tonic-gate 	char lpfile[BUFSIZ];
9840Sstevel@tonic-gate 
9850Sstevel@tonic-gate 	if ((svc == NULL) || (printer == NULL) || (job_id < 0) ||
9860Sstevel@tonic-gate 	    (attributes == NULL))
9870Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
9880Sstevel@tonic-gate 
9890Sstevel@tonic-gate 	if ((status = authorized(svc, job_id)) != PAPI_OK)
9900Sstevel@tonic-gate 		return (status);
9910Sstevel@tonic-gate 
9920Sstevel@tonic-gate 	if ((*job = j = calloc(1, sizeof (*j))) == NULL)
9930Sstevel@tonic-gate 		return (PAPI_TEMPORARY_ERROR);
9940Sstevel@tonic-gate 
9950Sstevel@tonic-gate 	dest = printer_name_from_uri_id(printer, job_id);
9960Sstevel@tonic-gate 	status = lpsched_start_change(svc, dest, job_id, &file);
9970Sstevel@tonic-gate 	if (status != PAPI_OK)
9980Sstevel@tonic-gate 		return (status);
9990Sstevel@tonic-gate 
10000Sstevel@tonic-gate 	if ((r = getrequest(file)) != NULL) {
10010Sstevel@tonic-gate 		job_attributes_to_lpsched_request(handle, r,
10020Sstevel@tonic-gate 				(papi_attribute_t **)attributes);
10030Sstevel@tonic-gate #ifdef LP_USE_PAPI_ATTR
10040Sstevel@tonic-gate 		/*
10050Sstevel@tonic-gate 		 * store the job attributes in the PAPI job attribute file
10060Sstevel@tonic-gate 		 * that was created by the origonal job request. We need to
10070Sstevel@tonic-gate 		 * modify the attributes in the file as per the new attributes
10080Sstevel@tonic-gate 		 */
10090Sstevel@tonic-gate 		snprintf(lpfile, sizeof (lpfile), "%s%d-%s",
10100Sstevel@tonic-gate 			"/var/spool/lp/temp/", job_id, LP_PAPIATTRNAME);
10110Sstevel@tonic-gate 		status = psm_modifyAttrsFile(attributes, lpfile);
10120Sstevel@tonic-gate 		if (status != PAPI_OK) {
10130Sstevel@tonic-gate 			detailed_error(svc,
10140Sstevel@tonic-gate 				"unable to modify the attributes file: %s: %s",
10150Sstevel@tonic-gate 				lpfile, strerror(errno));
10160Sstevel@tonic-gate 			return (PAPI_DEVICE_ERROR);
10170Sstevel@tonic-gate 		}
10180Sstevel@tonic-gate #endif
10190Sstevel@tonic-gate 
10200Sstevel@tonic-gate 		if (putrequest(file, r) < 0) {
10210Sstevel@tonic-gate 			detailed_error(svc,
10220Sstevel@tonic-gate 				gettext("failed to write job: %s: %s"),
10230Sstevel@tonic-gate 				file, strerror(errno));
10240Sstevel@tonic-gate 			freerequest(r);
10250Sstevel@tonic-gate 			return (PAPI_DEVICE_ERROR);
10260Sstevel@tonic-gate 		}
10270Sstevel@tonic-gate 	} else {
10280Sstevel@tonic-gate 		detailed_error(svc, gettext("failed to read job: %s: %s"),
10290Sstevel@tonic-gate 				file, strerror(errno));
10300Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
10310Sstevel@tonic-gate 	}
10320Sstevel@tonic-gate 
10330Sstevel@tonic-gate 	status = lpsched_end_change(svc, dest, job_id);
10340Sstevel@tonic-gate 	lpsched_request_to_job_attributes(r, j);
10350Sstevel@tonic-gate 	freerequest(r);
10360Sstevel@tonic-gate 
10370Sstevel@tonic-gate 	return (status);
10380Sstevel@tonic-gate }
10390Sstevel@tonic-gate 
10400Sstevel@tonic-gate /*
10410Sstevel@tonic-gate  * Extension to PAPI, a variation of this is slated for post-1.0
10420Sstevel@tonic-gate  */
10430Sstevel@tonic-gate #define	DUMMY_FILE	"/var/spool/lp/fifos/FIFO"
10440Sstevel@tonic-gate 
10450Sstevel@tonic-gate papi_status_t
10462264Sjacobs papiJobCreate(papi_service_t handle, char *printer,
10472264Sjacobs 		papi_attribute_t **job_attributes,
10482264Sjacobs 		papi_job_ticket_t *job_ticket, papi_job_t *job)
10490Sstevel@tonic-gate {
10500Sstevel@tonic-gate 	papi_status_t status;
10510Sstevel@tonic-gate 	service_t *svc = handle;
10520Sstevel@tonic-gate 	job_t *j = NULL;
10530Sstevel@tonic-gate 	REQUEST *request;
10540Sstevel@tonic-gate 	char *request_id = NULL;
10550Sstevel@tonic-gate 	char *c;
10560Sstevel@tonic-gate 	char *tmp = NULL;
10570Sstevel@tonic-gate 	char metadata_file[MAXPATHLEN];
10580Sstevel@tonic-gate 
10590Sstevel@tonic-gate 	if ((svc == NULL) || (printer == NULL) || (job == NULL))
10600Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
10610Sstevel@tonic-gate 
10620Sstevel@tonic-gate 	if (job_ticket != NULL)
10630Sstevel@tonic-gate 		return (PAPI_JOB_TICKET_NOT_SUPPORTED);
10640Sstevel@tonic-gate 
10650Sstevel@tonic-gate 	if ((*job = j = calloc(1, sizeof (*j))) == NULL)
10660Sstevel@tonic-gate 		return (PAPI_TEMPORARY_ERROR);
10670Sstevel@tonic-gate 
10680Sstevel@tonic-gate 	/* 1 for the control file (-0) */
10690Sstevel@tonic-gate 	status = lpsched_alloc_files(svc, 1, &request_id);
10700Sstevel@tonic-gate 	if (status != PAPI_OK)
10710Sstevel@tonic-gate 		return (status);
10720Sstevel@tonic-gate 
10730Sstevel@tonic-gate 	/* convert the attributes to an lpsched REQUEST structure */
10740Sstevel@tonic-gate 	request = create_request(svc, (char *)printer,
10750Sstevel@tonic-gate 				(papi_attribute_t **)job_attributes);
10762264Sjacobs 	if (request == NULL)
10772264Sjacobs 		return (PAPI_TEMPORARY_ERROR);
10780Sstevel@tonic-gate 	addlist(&request->file_list, DUMMY_FILE);	/* add a dummy file */
10790Sstevel@tonic-gate 	request->actions |= ACT_HOLD;			/* hold the job */
10800Sstevel@tonic-gate 
10810Sstevel@tonic-gate #ifdef LP_USE_PAPI_ATTR
10820Sstevel@tonic-gate 	/*
10830Sstevel@tonic-gate 	 * store the job attributes in the PAPI job attribute file that was
10840Sstevel@tonic-gate 	 * created by lpsched_alloc_files(), the attributes will then pass
10850Sstevel@tonic-gate 	 * through lpsched and be given to the slow-filters and the printer's
10860Sstevel@tonic-gate 	 * interface script to process them
10870Sstevel@tonic-gate 	 */
10880Sstevel@tonic-gate 	snprintf(metadata_file, sizeof (metadata_file), "%s%s-%s",
10890Sstevel@tonic-gate 		"/var/spool/lp/temp/", request_id, LP_PAPIATTRNAME);
10900Sstevel@tonic-gate 	status = psm_copy_attrsToFile(job_attributes, metadata_file);
10910Sstevel@tonic-gate 	if (status != PAPI_OK) {
10920Sstevel@tonic-gate 		detailed_error(svc, "unable to copy attributes to file: %s: %s",
10930Sstevel@tonic-gate 				metadata_file, strerror(errno));
10942264Sjacobs 		free(request_id);
10950Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
10960Sstevel@tonic-gate 	}
10970Sstevel@tonic-gate #endif
10980Sstevel@tonic-gate 
10990Sstevel@tonic-gate 	/* store the REQUEST on disk */
11000Sstevel@tonic-gate 	snprintf(metadata_file, sizeof (metadata_file), "%s-0", request_id);
11012264Sjacobs 	free(request_id);
11020Sstevel@tonic-gate 	if (putrequest(metadata_file, request) < 0) {
11030Sstevel@tonic-gate 		detailed_error(svc, gettext("unable to save request: %s: %s"),
11040Sstevel@tonic-gate 			metadata_file, strerror(errno));
11050Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
11060Sstevel@tonic-gate 	}
11070Sstevel@tonic-gate 
11080Sstevel@tonic-gate 	status = lpsched_commit_job(svc, metadata_file, &tmp);
11090Sstevel@tonic-gate 	if (status != PAPI_OK) {
11100Sstevel@tonic-gate 		unlink(metadata_file);
11110Sstevel@tonic-gate 		return (status);
11120Sstevel@tonic-gate 	}
11130Sstevel@tonic-gate 
11140Sstevel@tonic-gate 	lpsched_request_to_job_attributes(request, j);
11150Sstevel@tonic-gate 
11160Sstevel@tonic-gate 	if ((c = strrchr(tmp, '-')) != NULL)
11170Sstevel@tonic-gate 		c++;
11180Sstevel@tonic-gate 	papiAttributeListAddInteger(&j->attributes, PAPI_ATTR_REPLACE,
11190Sstevel@tonic-gate 			"job-id", atoi(c));
11200Sstevel@tonic-gate 	papiAttributeListAddString(&j->attributes, PAPI_ATTR_REPLACE,
11210Sstevel@tonic-gate 			"job-uri", tmp);
11220Sstevel@tonic-gate 
11230Sstevel@tonic-gate 	return (PAPI_OK);
11240Sstevel@tonic-gate }
11250Sstevel@tonic-gate 
11260Sstevel@tonic-gate papi_status_t
11270Sstevel@tonic-gate papiJobCommit(papi_service_t handle, char *printer, int32_t id)
11280Sstevel@tonic-gate {
11290Sstevel@tonic-gate 	papi_status_t status = PAPI_OK;
11300Sstevel@tonic-gate 	service_t *svc = handle;
11310Sstevel@tonic-gate 	REQUEST *r = NULL;
11320Sstevel@tonic-gate 	char *metadata_file;
11330Sstevel@tonic-gate 	char *dest;
11340Sstevel@tonic-gate 
11350Sstevel@tonic-gate 	if ((svc == NULL) || (printer == NULL))
11360Sstevel@tonic-gate 		return (PAPI_BAD_ARGUMENT);
11370Sstevel@tonic-gate 
11380Sstevel@tonic-gate 	dest = printer_name_from_uri_id(printer, id);
11390Sstevel@tonic-gate 	/* tell the scheduler that we want to change the job */
11400Sstevel@tonic-gate 	status = lpsched_start_change(svc, dest, id, &metadata_file);
11410Sstevel@tonic-gate 	if (status != PAPI_OK)
11420Sstevel@tonic-gate 		return (status);
11430Sstevel@tonic-gate 
11440Sstevel@tonic-gate 	if ((r = getrequest(metadata_file)) != NULL) {
11450Sstevel@tonic-gate 		r->actions &= ~ACT_RESUME;
11460Sstevel@tonic-gate 		r->actions |= ACT_RESUME;
11470Sstevel@tonic-gate 		dellist(&r->file_list, DUMMY_FILE);
11480Sstevel@tonic-gate 
11490Sstevel@tonic-gate 		if (putrequest(metadata_file, r) < 0) {
11500Sstevel@tonic-gate 			detailed_error(svc,
11510Sstevel@tonic-gate 				gettext("failed to write job: %s: %s"),
11520Sstevel@tonic-gate 				metadata_file, strerror(errno));
11532264Sjacobs 			freerequest(r);
11540Sstevel@tonic-gate 			return (PAPI_DEVICE_ERROR);
11550Sstevel@tonic-gate 		}
11560Sstevel@tonic-gate 	} else {
11570Sstevel@tonic-gate 		detailed_error(svc, gettext("failed to read job: %s: %s"),
11580Sstevel@tonic-gate 				metadata_file, strerror(errno));
11590Sstevel@tonic-gate 		return (PAPI_DEVICE_ERROR);
11600Sstevel@tonic-gate 	}
11610Sstevel@tonic-gate 
11620Sstevel@tonic-gate 	status = lpsched_end_change(svc, dest, id);
11630Sstevel@tonic-gate 	freerequest(r);
11640Sstevel@tonic-gate 
11650Sstevel@tonic-gate 	return (status);
11660Sstevel@tonic-gate }
11670Sstevel@tonic-gate 
11680Sstevel@tonic-gate papi_status_t
11690Sstevel@tonic-gate papiJobStreamAdd(papi_service_t handle, char *printer, int32_t id,
11700Sstevel@tonic-gate 		papi_stream_t *stream)
11710Sstevel@tonic-gate {
11720Sstevel@tonic-gate 	papi_status_t status;
11730Sstevel@tonic-gate 	service_t *svc = handle;
11740Sstevel@tonic-gate 	job_stream_t *s = NULL;
11750Sstevel@tonic-gate 	char *metadata_file = NULL;
11760Sstevel@tonic-gate 	char *dest;
11770Sstevel@tonic-gate 	char path[MAXPATHLEN];
11780Sstevel@tonic-gate 
11790Sstevel@tonic-gate 	/* allocate space for the stream */
11800Sstevel@tonic-gate 	if ((*stream = s = calloc(1, sizeof (*s))) == NULL)
11810Sstevel@tonic-gate 		return (PAPI_TEMPORARY_ERROR);
11820Sstevel@tonic-gate 
11830Sstevel@tonic-gate 	dest = printer_name_from_uri_id(printer, id);
11840Sstevel@tonic-gate 	/* create/open data file (only root or lp can really do this */
11850Sstevel@tonic-gate 	snprintf(path, sizeof (path), "/var/spool/lp/temp/%d-XXXXXX", id);
11860Sstevel@tonic-gate 	if ((s->fd = mkstemp(path)) < 0) {
11870Sstevel@tonic-gate 		detailed_error(svc, gettext("unable to create sink (%s): %s"),
11880Sstevel@tonic-gate 			path, strerror(errno));
11890Sstevel@tonic-gate 		free(s);
11900Sstevel@tonic-gate 		return (PAPI_NOT_AUTHORIZED);
11910Sstevel@tonic-gate 	}
11920Sstevel@tonic-gate 
11930Sstevel@tonic-gate 	/* add data file to job */
11940Sstevel@tonic-gate 	status = lpsched_start_change(svc, dest, id, &metadata_file);
11950Sstevel@tonic-gate 	if (status != PAPI_OK) {
11960Sstevel@tonic-gate 		close(s->fd);
11970Sstevel@tonic-gate 		free(s);
11980Sstevel@tonic-gate 		unlink(path);
11990Sstevel@tonic-gate 		return (status);
12000Sstevel@tonic-gate 	}
12010Sstevel@tonic-gate 
12020Sstevel@tonic-gate 	if ((s->request = getrequest(metadata_file)) == NULL) {
12030Sstevel@tonic-gate 		detailed_error(svc, gettext("unable to load request: %s: %s"),
12040Sstevel@tonic-gate 			metadata_file, strerror(errno));
12050Sstevel@tonic-gate 		close(s->fd);
12060Sstevel@tonic-gate 		free(s);
12070Sstevel@tonic-gate 		unlink(path);
12080Sstevel@tonic-gate 		return (PAPI_NOT_POSSIBLE);
12090Sstevel@tonic-gate 	}
12100Sstevel@tonic-gate 
12110Sstevel@tonic-gate 	addlist(&(s->request->file_list), path);
12120Sstevel@tonic-gate 
12130Sstevel@tonic-gate 	if (putrequest(metadata_file, s->request) < 0) {
12140Sstevel@tonic-gate 		detailed_error(svc, gettext("unable to save request: %s: %s"),
12150Sstevel@tonic-gate 			metadata_file, strerror(errno));
12160Sstevel@tonic-gate 		close(s->fd);
12170Sstevel@tonic-gate 		free(s);
12180Sstevel@tonic-gate 		unlink(path);
12190Sstevel@tonic-gate 		return (PAPI_NOT_POSSIBLE);
12200Sstevel@tonic-gate 	}
12210Sstevel@tonic-gate 
12220Sstevel@tonic-gate 	status = lpsched_end_change(svc, dest, id);
12230Sstevel@tonic-gate 
12240Sstevel@tonic-gate 	if (status != PAPI_OK)
12250Sstevel@tonic-gate 		return (status);
12260Sstevel@tonic-gate 
12270Sstevel@tonic-gate 	return (PAPI_OK);
12280Sstevel@tonic-gate }
1229