2013-12-03 22:21:44 -05:00
|
|
|
/*
|
|
|
|
* DNP DS40/DS80 Photo Printer CUPS backend -- libusb-1.0 version
|
|
|
|
*
|
2015-01-05 21:39:22 -05:00
|
|
|
* (c) 2013-2015 Solomon Peachy <pizza@shaftnet.org>
|
2013-12-03 22:21:44 -05:00
|
|
|
*
|
2013-12-17 08:31:45 -05:00
|
|
|
* Development of this backend was sponsored by:
|
|
|
|
*
|
2014-06-03 20:43:31 -04:00
|
|
|
* Marco Di Antonio and [ ilgruppodigitale.com ]
|
|
|
|
* LiveLink Technology [ www.livelinktechnology.net ]
|
2013-12-17 08:31:45 -05:00
|
|
|
*
|
2013-12-03 22:21:44 -05:00
|
|
|
* The latest version of this program can be found at:
|
2013-12-10 08:54:07 -05:00
|
|
|
*
|
2013-12-03 22:21:44 -05:00
|
|
|
* http://git.shaftnet.org/cgit/selphy_print.git
|
2013-12-10 08:54:07 -05:00
|
|
|
*
|
2013-12-03 22:21:44 -05:00
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License as published by the Free
|
|
|
|
* Software Foundation; either version 3 of the License, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
|
|
|
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* [http://www.gnu.org/licenses/gpl-3.0.html]
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <signal.h>
|
|
|
|
|
|
|
|
#include "backend_common.h"
|
|
|
|
|
|
|
|
#define USB_VID_DNP 0x1343
|
2014-02-04 19:24:20 -05:00
|
|
|
#define USB_PID_DNP_DS40 0x0003 // Also Citizen CX
|
2014-12-03 00:10:53 -05:00
|
|
|
#define USB_PID_DNP_DS80 0x0004 // Also Citizen CX-W, and Mitsubishi CP-3800DW
|
2014-02-04 19:24:20 -05:00
|
|
|
#define USB_PID_DNP_DSRX1 0x0005 // Also Citizen CY
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2015-06-10 23:37:27 -04:00
|
|
|
//#define USB_PID_DNP_DS620 XXXX
|
|
|
|
//#define USB_PID_DNP_DS80DX XXXX
|
2014-10-05 18:57:26 -04:00
|
|
|
//#define USB_PID_OLMEC_OP900 XXXX
|
2014-01-31 19:55:47 -05:00
|
|
|
//#define USB_PID_CITIZEN_CW-02 XXXXX
|
|
|
|
//#define USB_PID_CITIZEN_OP900II XXXXX
|
|
|
|
|
2013-12-03 22:21:44 -05:00
|
|
|
/* Private data stucture */
|
|
|
|
struct dnpds40_ctx {
|
|
|
|
struct libusb_device_handle *dev;
|
|
|
|
uint8_t endp_up;
|
|
|
|
uint8_t endp_down;
|
|
|
|
|
2014-02-02 14:11:17 -05:00
|
|
|
int type;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-17 00:36:06 -05:00
|
|
|
int buf_needed;
|
2015-06-08 08:06:18 -04:00
|
|
|
int last_matte;
|
2015-06-08 23:17:12 -04:00
|
|
|
|
2015-06-08 08:06:18 -04:00
|
|
|
uint32_t multicut;
|
2015-06-08 23:17:12 -04:00
|
|
|
int matte;
|
2014-02-02 14:17:12 -05:00
|
|
|
|
2013-12-17 00:36:06 -05:00
|
|
|
uint8_t *qty_offset;
|
2015-06-11 08:40:31 -04:00
|
|
|
uint8_t *buffctrl_offset;
|
2013-12-17 00:36:06 -05:00
|
|
|
|
2013-12-03 22:21:44 -05:00
|
|
|
uint8_t *databuf;
|
|
|
|
int datalen;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct dnpds40_cmd {
|
2013-12-16 20:15:48 -05:00
|
|
|
uint8_t esc; /* Fixed at ascii ESC, aka 0x1B */
|
|
|
|
uint8_t p; /* Fixed at ascii 'P' aka 0x50 */
|
2013-12-03 22:21:44 -05:00
|
|
|
uint8_t arg1[6];
|
|
|
|
uint8_t arg2[16];
|
|
|
|
uint8_t arg3[8]; /* Decimal value of arg4's length, or empty */
|
2013-12-08 07:14:27 -05:00
|
|
|
uint8_t arg4[0]; /* Extra payload if arg3 is non-empty
|
|
|
|
Doesn't have to be sent in the same URB */
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-08 07:14:27 -05:00
|
|
|
/* All unused elements are set to 0x20 (ie ascii space) */
|
2013-12-03 22:21:44 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
#define min(__x, __y) ((__x) < (__y)) ? __x : __y
|
|
|
|
|
|
|
|
static void dnpds40_build_cmd(struct dnpds40_cmd *cmd, char *arg1, char *arg2, uint32_t arg3_len)
|
|
|
|
{
|
|
|
|
memset(cmd, 0x20, sizeof(*cmd));
|
|
|
|
cmd->esc = 0x1b;
|
|
|
|
cmd->p = 0x50;
|
|
|
|
memcpy(cmd->arg1, arg1, min(strlen(arg1), sizeof(cmd->arg1)));
|
|
|
|
memcpy(cmd->arg2, arg2, min(strlen(arg2), sizeof(cmd->arg2)));
|
2013-12-16 20:15:48 -05:00
|
|
|
if (arg3_len) {
|
2014-02-11 13:41:15 -05:00
|
|
|
char buf[9];
|
2014-02-11 20:11:33 -05:00
|
|
|
snprintf(buf, sizeof(buf), "%08u", arg3_len);
|
2013-12-16 20:15:48 -05:00
|
|
|
memcpy(cmd->arg3, buf, 8);
|
|
|
|
}
|
2013-12-04 07:37:11 -05:00
|
|
|
|
2013-12-03 22:21:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dnpds40_cleanup_string(char *start, int len)
|
|
|
|
{
|
|
|
|
char *ptr = strchr(start, 0x0d);
|
|
|
|
|
2014-02-11 20:49:08 -05:00
|
|
|
if (ptr && (ptr - start < len)) {
|
2013-12-03 22:21:44 -05:00
|
|
|
*ptr = 0x00; /* If there is a <CR>, terminate there */
|
2014-02-11 20:49:08 -05:00
|
|
|
len = ptr - start;
|
|
|
|
} else {
|
|
|
|
start[--len] = 0x00; /* force null-termination */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Trim trailing spaces */
|
|
|
|
while (len && start[len-1] == ' ') {
|
|
|
|
start[--len] = 0;
|
|
|
|
}
|
2013-12-03 22:21:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static char *dnpds40_media_types(char *str)
|
|
|
|
{
|
|
|
|
char tmp[4];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
memcpy(tmp, str + 4, 3);
|
|
|
|
tmp[3] = 0;
|
|
|
|
|
|
|
|
i = atoi(tmp);
|
|
|
|
|
2015-06-08 08:06:18 -04:00
|
|
|
/* Subtract out the "mark" type */
|
|
|
|
if (i & 1)
|
|
|
|
i--;
|
|
|
|
|
2013-12-03 22:21:44 -05:00
|
|
|
switch (i) {
|
|
|
|
case 200: return "5x3.5 (L)";
|
|
|
|
case 210: return "5x7 (2L)";
|
|
|
|
case 300: return "6x4 (PC)";
|
|
|
|
case 310: return "6x8 (A5)";
|
|
|
|
case 400: return "6x9 (A5W)";
|
|
|
|
case 500: return "8x10";
|
|
|
|
case 510: return "8x12";
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return "Unknown type";
|
|
|
|
}
|
|
|
|
|
2013-12-03 22:32:08 -05:00
|
|
|
static char *dnpds40_statuses(char *str)
|
|
|
|
{
|
|
|
|
char tmp[6];
|
|
|
|
int i;
|
|
|
|
memcpy(tmp, str, 5);
|
|
|
|
tmp[5] = 0;
|
|
|
|
|
|
|
|
i = atoi(tmp);
|
2013-12-10 08:54:07 -05:00
|
|
|
|
2013-12-03 22:32:08 -05:00
|
|
|
switch (i) {
|
|
|
|
case 0: return "Idle";
|
|
|
|
case 1: return "Printing";
|
|
|
|
case 500: return "Cooling Print Head";
|
|
|
|
case 510: return "Cooling Paper Motor";
|
|
|
|
case 1000: return "Cover Open";
|
|
|
|
case 1010: return "No Scrap Box";
|
|
|
|
case 1100: return "Paper End";
|
|
|
|
case 1200: return "Ribbon End";
|
2014-02-02 14:11:17 -05:00
|
|
|
case 1300: return "Paper Jam";
|
|
|
|
case 1400: return "Ribbon Error";
|
2013-12-03 22:32:08 -05:00
|
|
|
case 1500: return "Paper Definition Error";
|
|
|
|
case 1600: return "Data Error";
|
|
|
|
case 2000: return "Head Voltage Error";
|
|
|
|
case 2100: return "Head Position Error";
|
|
|
|
case 2200: return "Power Supply Fan Error";
|
|
|
|
case 2300: return "Cutter Error";
|
|
|
|
case 2400: return "Pinch Roller Error";
|
|
|
|
case 2500: return "Abnormal Head Temperature";
|
|
|
|
case 2600: return "Abnormal Media Temperature";
|
|
|
|
case 2610: return "Abnormal Paper Motor Temperature";
|
|
|
|
case 2700: return "Ribbon Tension Error";
|
|
|
|
case 2800: return "RF-ID Module Error";
|
|
|
|
case 3000: return "System Error";
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-02-02 14:11:17 -05:00
|
|
|
return "Unkown Error";
|
2013-12-03 22:32:08 -05:00
|
|
|
}
|
|
|
|
|
2013-12-10 21:53:46 -05:00
|
|
|
static int dnpds40_do_cmd(struct dnpds40_ctx *ctx,
|
2013-12-11 07:50:31 -05:00
|
|
|
struct dnpds40_cmd *cmd,
|
|
|
|
uint8_t *data, int len)
|
2013-12-10 21:53:46 -05:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if ((ret = send_data(ctx->dev, ctx->endp_down,
|
|
|
|
(uint8_t*)cmd, sizeof(*cmd))))
|
|
|
|
return ret;
|
|
|
|
|
2013-12-11 07:50:31 -05:00
|
|
|
if (data && len)
|
2013-12-10 21:53:46 -05:00
|
|
|
if ((ret = send_data(ctx->dev, ctx->endp_down,
|
|
|
|
data, len)))
|
|
|
|
return ret;
|
|
|
|
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_OK;
|
2013-12-10 21:53:46 -05:00
|
|
|
}
|
|
|
|
|
2013-12-04 10:31:41 -05:00
|
|
|
static uint8_t * dnpds40_resp_cmd(struct dnpds40_ctx *ctx,
|
|
|
|
struct dnpds40_cmd *cmd,
|
|
|
|
int *len)
|
|
|
|
{
|
|
|
|
char tmp[9];
|
|
|
|
uint8_t *respbuf;
|
|
|
|
|
|
|
|
int ret, i, num = 0;
|
2013-12-08 07:14:27 -05:00
|
|
|
|
|
|
|
memset(tmp, 0, sizeof(tmp));
|
|
|
|
|
2013-12-10 21:53:46 -05:00
|
|
|
if ((ret = dnpds40_do_cmd(ctx, cmd, NULL, 0)))
|
2013-12-04 10:31:41 -05:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* Read in the response header */
|
2013-12-21 22:55:33 -05:00
|
|
|
ret = read_data(ctx->dev, ctx->endp_up,
|
|
|
|
(uint8_t*)tmp, 8, &num);
|
|
|
|
if (ret < 0)
|
2013-12-04 10:31:41 -05:00
|
|
|
return NULL;
|
|
|
|
|
2013-12-21 22:55:33 -05:00
|
|
|
if (num != 8) {
|
|
|
|
ERROR("Short read! (%d/%d)\n", num, 8);
|
|
|
|
return NULL;
|
2013-12-11 07:50:31 -05:00
|
|
|
}
|
2013-12-04 10:31:41 -05:00
|
|
|
|
2013-12-11 07:50:31 -05:00
|
|
|
i = atoi(tmp); /* Length of payload in bytes, possibly padded */
|
2013-12-04 10:31:41 -05:00
|
|
|
respbuf = malloc(i);
|
|
|
|
|
|
|
|
/* Read in the actual response */
|
2013-12-21 22:55:33 -05:00
|
|
|
ret = read_data(ctx->dev, ctx->endp_up,
|
|
|
|
respbuf, i, &num);
|
|
|
|
if (ret < 0) {
|
|
|
|
free(respbuf);
|
|
|
|
return NULL;
|
2013-12-11 07:50:31 -05:00
|
|
|
}
|
|
|
|
|
2013-12-21 22:55:33 -05:00
|
|
|
if (num != i) {
|
|
|
|
ERROR("Short read! (%d/%d)\n", num, i);
|
2013-12-04 10:31:41 -05:00
|
|
|
free(respbuf);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*len = num;
|
|
|
|
return respbuf;
|
|
|
|
}
|
|
|
|
|
2013-12-16 20:15:48 -05:00
|
|
|
static int dnpds40_query_serno(struct libusb_device_handle *dev, uint8_t endp_up, uint8_t endp_down, char *buf, int buf_len)
|
|
|
|
{
|
|
|
|
struct dnpds40_cmd cmd;
|
|
|
|
uint8_t *resp;
|
|
|
|
int len = 0;
|
|
|
|
|
|
|
|
struct dnpds40_ctx ctx = {
|
|
|
|
.dev = dev,
|
|
|
|
.endp_up = endp_up,
|
|
|
|
.endp_down = endp_down,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Get Serial Number */
|
|
|
|
dnpds40_build_cmd(&cmd, "INFO", "SERIAL_NUMBER", 0);
|
|
|
|
|
|
|
|
resp = dnpds40_resp_cmd(&ctx, &cmd, &len);
|
|
|
|
if (!resp)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-16 20:15:48 -05:00
|
|
|
|
|
|
|
dnpds40_cleanup_string((char*)resp, len);
|
|
|
|
|
|
|
|
strncpy(buf, (char*)resp, buf_len);
|
|
|
|
buf[buf_len-1] = 0;
|
|
|
|
|
|
|
|
free(resp);
|
|
|
|
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_OK;
|
2013-12-16 20:15:48 -05:00
|
|
|
}
|
|
|
|
|
2013-12-03 22:21:44 -05:00
|
|
|
static void *dnpds40_init(void)
|
|
|
|
{
|
|
|
|
struct dnpds40_ctx *ctx = malloc(sizeof(struct dnpds40_ctx));
|
|
|
|
if (!ctx)
|
|
|
|
return NULL;
|
|
|
|
memset(ctx, 0, sizeof(struct dnpds40_ctx));
|
|
|
|
|
|
|
|
ctx->type = P_ANY;
|
2015-06-08 08:06:18 -04:00
|
|
|
ctx->last_matte = -1;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
|
|
|
return ctx;
|
|
|
|
}
|
|
|
|
|
2013-12-10 08:54:07 -05:00
|
|
|
static void dnpds40_attach(void *vctx, struct libusb_device_handle *dev,
|
2015-06-11 08:40:31 -04:00
|
|
|
uint8_t endp_up, uint8_t endp_down, uint8_t jobid)
|
2013-12-03 22:21:44 -05:00
|
|
|
{
|
|
|
|
struct dnpds40_ctx *ctx = vctx;
|
|
|
|
struct libusb_device *device;
|
|
|
|
struct libusb_device_descriptor desc;
|
|
|
|
|
|
|
|
UNUSED(jobid);
|
|
|
|
|
2013-12-10 08:54:07 -05:00
|
|
|
ctx->dev = dev;
|
2013-12-03 22:21:44 -05:00
|
|
|
ctx->endp_up = endp_up;
|
|
|
|
ctx->endp_down = endp_down;
|
|
|
|
|
|
|
|
device = libusb_get_device(dev);
|
|
|
|
libusb_get_device_descriptor(device, &desc);
|
|
|
|
|
|
|
|
/* Map out device type */
|
|
|
|
if (desc.idProduct == USB_PID_DNP_DS40)
|
|
|
|
ctx->type = P_DNP_DS40;
|
|
|
|
else
|
|
|
|
ctx->type = P_DNP_DS80;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dnpds40_teardown(void *vctx) {
|
|
|
|
struct dnpds40_ctx *ctx = vctx;
|
|
|
|
|
|
|
|
if (!ctx)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (ctx->databuf)
|
|
|
|
free(ctx->databuf);
|
|
|
|
free(ctx);
|
|
|
|
}
|
|
|
|
|
2013-12-17 20:43:30 -05:00
|
|
|
#define MAX_PRINTJOB_LEN (((2448*7536+1024+54))*3+1024) /* Worst-case */
|
2013-12-03 22:21:44 -05:00
|
|
|
|
|
|
|
static int dnpds40_read_parse(void *vctx, int data_fd) {
|
|
|
|
struct dnpds40_ctx *ctx = vctx;
|
2014-02-11 13:41:15 -05:00
|
|
|
int run = 1;
|
2013-12-17 00:36:06 -05:00
|
|
|
char buf[9] = { 0 };
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2015-05-31 18:00:39 -04:00
|
|
|
uint32_t matte, multicut, dpi;
|
2014-02-02 14:11:17 -05:00
|
|
|
|
2013-12-03 22:21:44 -05:00
|
|
|
if (!ctx)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2014-01-21 20:34:00 -05:00
|
|
|
if (ctx->databuf) {
|
2014-01-19 19:23:24 -05:00
|
|
|
free(ctx->databuf);
|
2014-01-21 20:34:00 -05:00
|
|
|
ctx->databuf = NULL;
|
|
|
|
}
|
2014-01-19 19:23:24 -05:00
|
|
|
|
2014-02-02 14:11:17 -05:00
|
|
|
/* There's no way to figure out the total job length in advance, we
|
|
|
|
have to parse the stream until we get to the image plane data,
|
|
|
|
and even then the stream can contain arbitrary commands later.
|
|
|
|
|
|
|
|
So instead, we allocate a buffer of the maximum possible length,
|
|
|
|
then parse the incoming stream until we hit the START command at
|
|
|
|
the end of the job.
|
|
|
|
*/
|
|
|
|
|
2013-12-17 00:36:06 -05:00
|
|
|
ctx->datalen = 0;
|
2013-12-03 22:21:44 -05:00
|
|
|
ctx->databuf = malloc(MAX_PRINTJOB_LEN);
|
|
|
|
if (!ctx->databuf) {
|
|
|
|
ERROR("Memory allocation failure!\n");
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
2013-12-03 22:21:44 -05:00
|
|
|
}
|
|
|
|
|
2015-05-31 18:00:39 -04:00
|
|
|
/* Clear everything out */
|
|
|
|
matte = 0;
|
|
|
|
dpi = 0;
|
|
|
|
multicut = 0;
|
2015-06-11 08:40:31 -04:00
|
|
|
ctx->buffctrl_offset = ctx->qty_offset = 0;
|
2015-05-31 18:00:39 -04:00
|
|
|
|
2014-01-19 19:23:24 -05:00
|
|
|
while (run) {
|
2014-02-11 13:41:15 -05:00
|
|
|
int remain, i, j;
|
2014-02-02 14:11:17 -05:00
|
|
|
/* Read in command header */
|
2013-12-17 00:36:06 -05:00
|
|
|
i = read(data_fd, ctx->databuf + ctx->datalen,
|
|
|
|
sizeof(struct dnpds40_cmd));
|
|
|
|
if (i < 0)
|
|
|
|
return i;
|
|
|
|
if (i == 0)
|
|
|
|
break;
|
|
|
|
if (i < (int) sizeof(struct dnpds40_cmd))
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
2013-12-17 00:36:06 -05:00
|
|
|
|
|
|
|
if (ctx->databuf[ctx->datalen + 0] != 0x1b ||
|
|
|
|
ctx->databuf[ctx->datalen + 1] != 0x50) {
|
|
|
|
ERROR("Unrecognized header data format @%d!\n", ctx->datalen);
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
2013-12-17 00:36:06 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Parse out length of data chunk, if any */
|
|
|
|
memcpy(buf, ctx->databuf + ctx->datalen + 24, 8);
|
|
|
|
j = atoi(buf);
|
|
|
|
|
2013-12-19 07:56:16 -05:00
|
|
|
/* Read in data chunk as quickly as possible */
|
|
|
|
remain = j;
|
|
|
|
while (remain > 0) {
|
|
|
|
i = read(data_fd, ctx->databuf + ctx->datalen + sizeof(struct dnpds40_cmd),
|
|
|
|
remain);
|
|
|
|
if (i < 0)
|
|
|
|
return i;
|
|
|
|
if (i == 0)
|
|
|
|
return 1;
|
|
|
|
ctx->datalen += i;
|
|
|
|
remain -= i;
|
|
|
|
}
|
|
|
|
ctx->datalen -= j; /* Back it off */
|
2013-12-17 00:36:06 -05:00
|
|
|
|
|
|
|
/* Check for some offsets */
|
|
|
|
if(!memcmp("CNTRL QTY", ctx->databuf + ctx->datalen+2, 9)) {
|
|
|
|
ctx->qty_offset = ctx->databuf + ctx->datalen + 32;
|
|
|
|
}
|
2015-06-11 08:40:31 -04:00
|
|
|
if(!memcmp("CNTRL BUFFCNTRL", ctx->databuf + ctx->datalen+2, 15)) {
|
|
|
|
ctx->buffctrl_offset = ctx->databuf + ctx->datalen + 32;
|
|
|
|
}
|
2014-02-02 14:11:17 -05:00
|
|
|
if(!memcmp("CNTRL OVERCOAT", ctx->databuf + ctx->datalen+2, 14)) {
|
|
|
|
memcpy(buf, ctx->databuf + ctx->datalen + 32, 8);
|
|
|
|
matte = atoi(buf);
|
|
|
|
}
|
|
|
|
if(!memcmp("CNTRL MULTICUT", ctx->databuf + ctx->datalen+2, 14)) {
|
|
|
|
memcpy(buf, ctx->databuf + ctx->datalen + 32, 8);
|
|
|
|
multicut = atoi(buf);
|
|
|
|
}
|
2015-05-31 18:00:39 -04:00
|
|
|
if(!memcmp("IMAGE YPLANE", ctx->databuf + ctx->datalen + 2, 12)) {
|
|
|
|
uint32_t x_ppm; /* Pixels Per Meter */
|
|
|
|
|
2014-02-02 14:11:17 -05:00
|
|
|
memcpy(&x_ppm, ctx->databuf + ctx->datalen + 32 + 42, sizeof(x_ppm));
|
|
|
|
x_ppm = le32_to_cpu(x_ppm);
|
|
|
|
|
2015-05-31 18:00:39 -04:00
|
|
|
switch (x_ppm) {
|
|
|
|
case 11808:
|
|
|
|
dpi = 300;
|
|
|
|
break;
|
|
|
|
case 23615:
|
2014-02-02 14:11:17 -05:00
|
|
|
dpi = 600;
|
2015-05-31 18:00:39 -04:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
WARNING("Unrecognized resolution (%d ppm), assuming 300dpi\n", x_ppm);
|
|
|
|
dpi = 300;
|
|
|
|
break;
|
2013-12-17 00:36:06 -05:00
|
|
|
}
|
|
|
|
}
|
2013-12-10 21:53:46 -05:00
|
|
|
|
2014-01-19 19:23:24 -05:00
|
|
|
/* This is the last block.. */
|
|
|
|
if(!memcmp("CNTRL START", ctx->databuf + ctx->datalen + 2, 11))
|
|
|
|
run = 0;
|
|
|
|
|
2013-12-17 00:36:06 -05:00
|
|
|
/* Add in the size of this chunk */
|
|
|
|
ctx->datalen += sizeof(struct dnpds40_cmd) + j;
|
2013-12-03 22:21:44 -05:00
|
|
|
}
|
2014-02-02 14:17:12 -05:00
|
|
|
|
2015-05-31 18:00:39 -04:00
|
|
|
/* Figure out the number of buffers we need. Most only need one. */
|
2015-06-13 12:01:08 -04:00
|
|
|
if (multicut) {
|
|
|
|
ctx->buf_needed = 1;
|
|
|
|
|
|
|
|
if (dpi == 600) {
|
|
|
|
if (ctx->type == P_DNP_DS80) { /* DS80/CX-W */
|
|
|
|
if (matte && (multicut == 21 || // A4 length
|
|
|
|
multicut == 20 || // 8x4*3
|
|
|
|
multicut == 19 || // 8x8+8x4
|
|
|
|
multicut == 15 || // 8x6*2
|
|
|
|
multicut == 7)) // 8x12
|
|
|
|
ctx->buf_needed = 2;
|
|
|
|
} else { /* DS40/RX1/CX/CY/etc */
|
|
|
|
if (multicut == 4 || // 6x8
|
|
|
|
multicut == 5 || // 6x9
|
|
|
|
multicut == 12) // 6x4*2
|
|
|
|
ctx->buf_needed = 2;
|
|
|
|
else if (matte && multicut == 3) // 5x7
|
|
|
|
ctx->buf_needed = 2;
|
|
|
|
}
|
2015-05-31 18:00:39 -04:00
|
|
|
}
|
2015-06-13 12:01:08 -04:00
|
|
|
} else {
|
|
|
|
WARNING("Missing or illegal MULTICUT command, can't validate print job against loaded media!\n");
|
|
|
|
if (dpi == 300)
|
|
|
|
ctx->buf_needed = 1;
|
|
|
|
else
|
|
|
|
ctx->buf_needed = 2;
|
2015-05-31 18:00:39 -04:00
|
|
|
}
|
|
|
|
|
2015-06-08 08:06:18 -04:00
|
|
|
ctx->multicut = multicut;
|
2015-06-08 23:17:12 -04:00
|
|
|
ctx->matte = (int)matte;
|
2015-05-31 18:00:39 -04:00
|
|
|
|
2015-06-08 23:17:12 -04:00
|
|
|
DEBUG("dpi %u matte %u mcut %u bufs %d\n",
|
|
|
|
dpi, matte, multicut, ctx->buf_needed);
|
2014-02-02 14:17:12 -05:00
|
|
|
|
2014-01-20 19:41:52 -05:00
|
|
|
if (!ctx->datalen)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_OK;
|
2013-12-03 22:21:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static int dnpds40_main_loop(void *vctx, int copies) {
|
|
|
|
struct dnpds40_ctx *ctx = vctx;
|
|
|
|
int ret;
|
2013-12-10 21:53:46 -05:00
|
|
|
struct dnpds40_cmd cmd;
|
|
|
|
uint8_t *resp = NULL;
|
|
|
|
int len = 0;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-16 23:01:30 -05:00
|
|
|
uint8_t *ptr;
|
|
|
|
char buf[9];
|
|
|
|
|
2013-12-03 22:21:44 -05:00
|
|
|
if (!ctx)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2015-06-08 08:06:18 -04:00
|
|
|
/* Query Media Info */
|
|
|
|
dnpds40_build_cmd(&cmd, "INFO", "MEDIA", 0);
|
|
|
|
|
|
|
|
resp = dnpds40_resp_cmd(ctx, &cmd, &len);
|
|
|
|
if (!resp)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
|
|
|
dnpds40_cleanup_string((char*)resp, len);
|
|
|
|
|
|
|
|
/* Sanity-check media type vs loaded media */
|
2015-06-13 12:01:08 -04:00
|
|
|
if (ctx->multicut) {
|
2015-06-08 08:06:18 -04:00
|
|
|
char tmp[4];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
memcpy(tmp, resp + 4, 3);
|
|
|
|
tmp[3] = 0;
|
|
|
|
|
|
|
|
i = atoi(tmp);
|
|
|
|
|
|
|
|
/* Subtract out the "mark" type */
|
|
|
|
if (i & 1)
|
|
|
|
i--;
|
|
|
|
|
|
|
|
switch(i) {
|
|
|
|
case 200: //"5x3.5 (L)"
|
|
|
|
if (ctx->multicut != 1) {
|
|
|
|
ERROR("Incorrect media for job loaded (%d)\n", i);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 210: //"5x7 (2L)"
|
|
|
|
if (ctx->multicut != 1 && ctx->multicut != 3) {
|
|
|
|
ERROR("Incorrect media for job loaded (%d)\n", i);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 300: //"6x4 (PC)"
|
|
|
|
if (ctx->multicut != 2 && ctx->multicut != 4 && ctx->multicut != 5) {
|
|
|
|
ERROR("Incorrect media for job loaded (%d)\n", i);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 310: //"6x8 (A5)"
|
|
|
|
if (ctx->multicut != 4 && ctx->multicut != 5) {
|
|
|
|
ERROR("Incorrect media for job loaded (%d)\n", i);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 400: //"6x9 (A5W)"
|
|
|
|
if (ctx->multicut != 5) {
|
|
|
|
ERROR("Incorrect media for job loaded (%d)\n", i);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 500: //"8x10"
|
|
|
|
if (ctx->multicut < 6 ||
|
|
|
|
ctx->multicut == 7 || ctx->multicut == 15 ||
|
|
|
|
ctx->multicut >= 18 ) {
|
|
|
|
ERROR("Incorrect media for job loaded (%d)\n", i);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
case 510: //"8x12"
|
|
|
|
if (ctx->multicut < 6 || ctx->multicut > 21) {
|
|
|
|
ERROR("Incorrect media for job loaded (%d)\n", i);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ERROR("Unknown media (%d)!\n", i);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX check firmware version and a few other things
|
|
|
|
// eg RX1 doesn't handle 6x9 media/prints, only DS80 handles 8" prints
|
|
|
|
// 2x6 on RX1 requires FW1.10 or newer
|
|
|
|
// 2x6 on DS40 requires FW1.40 or newer
|
2015-06-10 23:37:27 -04:00
|
|
|
// 6x9 on DS620 requires FW??? or newer
|
2015-06-08 23:17:12 -04:00
|
|
|
// all matte-related features require FW1.30 on DS40/DS80
|
|
|
|
// BUFFCNTRL requires FW1.30 on DS40/DS80
|
|
|
|
|
2015-06-11 08:40:31 -04:00
|
|
|
/* Update quantity offset with count */
|
|
|
|
if (copies > 1) {
|
|
|
|
// XXX should we verify we have sufficient media for prints?
|
2015-06-08 23:17:12 -04:00
|
|
|
snprintf(buf, sizeof(buf), "%07d\r", copies);
|
2015-06-11 08:40:31 -04:00
|
|
|
if (ctx->qty_offset) {
|
|
|
|
memcpy(ctx->qty_offset, buf, 8);
|
|
|
|
} else {
|
|
|
|
dnpds40_build_cmd(&cmd, "CNTRL", "QTY", 8);
|
|
|
|
if ((ret = dnpds40_do_cmd(ctx, &cmd, (uint8_t*)buf, 8)))
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
}
|
2015-06-08 23:17:12 -04:00
|
|
|
|
2015-06-11 08:40:31 -04:00
|
|
|
copies = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable job resumption on correctable errors */
|
|
|
|
snprintf(buf, sizeof(buf), "%08d", 1);
|
|
|
|
if (ctx->buffctrl_offset) {
|
|
|
|
memcpy(ctx->qty_offset, buf, 8);
|
|
|
|
} else {
|
2015-06-08 23:17:12 -04:00
|
|
|
dnpds40_build_cmd(&cmd, "CNTRL", "BUFFCNTRL", 8);
|
|
|
|
if ((ret = dnpds40_do_cmd(ctx, &cmd, (uint8_t*)buf, 8)))
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check our current job's lamination vs previous job. */
|
|
|
|
// XXX load last_matte from a status file
|
|
|
|
if (ctx->matte != ctx->last_matte)
|
|
|
|
ctx->buf_needed = 2; /* Switching needs both buffers */
|
|
|
|
ctx->last_matte = ctx->matte; // XXX write to status file
|
2015-06-08 08:06:18 -04:00
|
|
|
|
2013-12-10 21:53:46 -05:00
|
|
|
top:
|
2013-12-08 07:14:27 -05:00
|
|
|
|
2013-12-10 21:53:46 -05:00
|
|
|
if (resp) free(resp);
|
2013-12-08 07:14:27 -05:00
|
|
|
|
2013-12-10 21:53:46 -05:00
|
|
|
/* Query status */
|
|
|
|
dnpds40_build_cmd(&cmd, "STATUS", "", 0);
|
|
|
|
resp = dnpds40_resp_cmd(ctx, &cmd, &len);
|
|
|
|
if (!resp)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-10 21:53:46 -05:00
|
|
|
dnpds40_cleanup_string((char*)resp, len);
|
|
|
|
|
|
|
|
/* If we're not idle */
|
|
|
|
if (strcmp("00000", (char*)resp)) {
|
2013-12-17 00:36:06 -05:00
|
|
|
if (!strcmp("00001", (char*)resp)) {
|
2015-06-08 23:17:12 -04:00
|
|
|
int bufs;
|
2015-05-31 18:00:39 -04:00
|
|
|
|
2013-12-17 00:36:06 -05:00
|
|
|
free(resp);
|
|
|
|
/* Query buffer state */
|
|
|
|
dnpds40_build_cmd(&cmd, "INFO", "FREE_PBUFFER", 0);
|
|
|
|
resp = dnpds40_resp_cmd(ctx, &cmd, &len);
|
|
|
|
if (!resp)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-17 00:36:06 -05:00
|
|
|
dnpds40_cleanup_string((char*)resp, len);
|
|
|
|
|
2014-02-02 14:11:17 -05:00
|
|
|
/* Check to see if we have sufficient buffers */
|
2015-06-08 23:17:12 -04:00
|
|
|
bufs = atoi(((char*)resp)+3);
|
|
|
|
if (bufs < ctx->buf_needed) {
|
|
|
|
INFO("Insufficient printer buffers (%d vs %d), retrying...\n", bufs, ctx->buf_needed);
|
2013-12-17 00:36:06 -05:00
|
|
|
sleep(1);
|
|
|
|
goto top;
|
|
|
|
}
|
|
|
|
} else if (!strcmp("00500", (char*)resp) ||
|
|
|
|
!strcmp("00510", (char*)resp)) {
|
2015-06-08 08:06:18 -04:00
|
|
|
INFO("Printer cooling down...\n");
|
2013-12-10 21:53:46 -05:00
|
|
|
sleep(1);
|
|
|
|
goto top;
|
2015-06-08 08:06:18 -04:00
|
|
|
} else if (!strcmp("01500", (char*)resp)) {
|
|
|
|
ERROR("Paper definition error, aborting job\n");
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
} else if (!strcmp("01600", (char*)resp)) {
|
|
|
|
ERROR("Data error, aborting job\n");
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
2014-10-29 22:07:58 -04:00
|
|
|
} else {
|
|
|
|
ERROR("Printer Status: %s => %s\n", (char*)resp, dnpds40_statuses((char*)resp));
|
|
|
|
free(resp);
|
|
|
|
return CUPS_BACKEND_RETRY_CURRENT;
|
2013-12-10 21:53:46 -05:00
|
|
|
}
|
2013-12-03 22:21:44 -05:00
|
|
|
}
|
2013-12-10 21:53:46 -05:00
|
|
|
|
2013-12-16 23:01:30 -05:00
|
|
|
/* Send the stream over as individual data chunks */
|
|
|
|
ptr = ctx->databuf;
|
2013-12-17 00:36:06 -05:00
|
|
|
|
2013-12-16 23:01:30 -05:00
|
|
|
while(ptr && ptr < (ctx->databuf + ctx->datalen)) {
|
|
|
|
int i;
|
2013-12-17 00:36:06 -05:00
|
|
|
buf[8] = 0;
|
2013-12-16 23:01:30 -05:00
|
|
|
memcpy(buf, ptr + 24, 8);
|
2013-12-17 00:36:06 -05:00
|
|
|
i = atoi(buf) + 32;
|
|
|
|
|
2013-12-16 23:01:30 -05:00
|
|
|
|
|
|
|
if ((ret = send_data(ctx->dev, ctx->endp_down,
|
|
|
|
ptr, i)))
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-16 23:01:30 -05:00
|
|
|
|
|
|
|
ptr += i;
|
|
|
|
}
|
2013-12-16 22:05:49 -05:00
|
|
|
|
2013-12-10 21:53:46 -05:00
|
|
|
/* Clean up */
|
|
|
|
if (terminate)
|
|
|
|
copies = 1;
|
|
|
|
|
2014-01-22 09:10:34 -05:00
|
|
|
INFO("Print complete (%d copies remaining)\n", copies - 1);
|
2013-12-10 21:53:46 -05:00
|
|
|
|
|
|
|
if (copies && --copies) {
|
|
|
|
goto top;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (resp) free(resp);
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_OK;
|
2013-12-03 22:21:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static int dnpds40_get_info(struct dnpds40_ctx *ctx)
|
|
|
|
{
|
|
|
|
struct dnpds40_cmd cmd;
|
2013-12-04 10:31:41 -05:00
|
|
|
uint8_t *resp;
|
|
|
|
int len = 0;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-16 20:15:48 -05:00
|
|
|
/* Get Serial Number */
|
|
|
|
dnpds40_build_cmd(&cmd, "INFO", "SERIAL_NUMBER", 0);
|
|
|
|
|
|
|
|
resp = dnpds40_resp_cmd(ctx, &cmd, &len);
|
|
|
|
if (!resp)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-16 20:15:48 -05:00
|
|
|
|
|
|
|
dnpds40_cleanup_string((char*)resp, len);
|
|
|
|
|
|
|
|
INFO("Serial Number: '%s'\n", (char*)resp);
|
|
|
|
|
|
|
|
free(resp);
|
|
|
|
|
2013-12-03 22:21:44 -05:00
|
|
|
/* Get Firmware Version */
|
|
|
|
dnpds40_build_cmd(&cmd, "INFO", "FVER", 0);
|
|
|
|
|
2013-12-04 10:31:41 -05:00
|
|
|
resp = dnpds40_resp_cmd(ctx, &cmd, &len);
|
|
|
|
if (!resp)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-04 10:31:41 -05:00
|
|
|
dnpds40_cleanup_string((char*)resp, len);
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-04 10:31:41 -05:00
|
|
|
INFO("Firmware Version: '%s'\n", (char*)resp);
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-04 10:31:41 -05:00
|
|
|
free(resp);
|
2013-12-03 22:21:44 -05:00
|
|
|
|
|
|
|
/* Get Sensor Info */
|
|
|
|
dnpds40_build_cmd(&cmd, "INFO", "SENSOR", 0);
|
|
|
|
|
2013-12-04 10:31:41 -05:00
|
|
|
resp = dnpds40_resp_cmd(ctx, &cmd, &len);
|
|
|
|
if (!resp)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-04 10:31:41 -05:00
|
|
|
dnpds40_cleanup_string((char*)resp, len);
|
2015-06-10 23:37:27 -04:00
|
|
|
INFO("Sensor Info:\n");
|
|
|
|
{
|
|
|
|
char *tmp;
|
|
|
|
tmp = strtok((char*)resp, "; ");
|
|
|
|
do {
|
|
|
|
// XXX parse the components?
|
|
|
|
INFO(" %s\n", tmp);
|
|
|
|
} while ((tmp = strtok(NULL, "; ")) != NULL);
|
|
|
|
}
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-04 10:31:41 -05:00
|
|
|
free(resp);
|
2013-12-03 22:21:44 -05:00
|
|
|
|
2013-12-16 20:15:48 -05:00
|
|
|
/* Get Qty of prints made on this media? */
|
|
|
|
dnpds40_build_cmd(&cmd, "INFO", "PQTY", 0);
|
|
|
|
|
|
|
|
resp = dnpds40_resp_cmd(ctx, &cmd, &len);
|
|
|
|
if (!resp)
|
2014-04-20 11:51:06 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-12-16 20:15:48 -05:00
|
|
|
|
|
|
|
dnpds40_cleanup_string((char*)resp, len);
|
|