2019-04-17 13:58:42 -04:00
|
|
|
/*
|
|
|
|
* Sony UP-D series (new) Photo Printer CUPS backend -- libusb-1.0 version
|
|
|
|
*
|
2020-06-30 23:42:54 -04:00
|
|
|
* (c) 2019-2020 Solomon Peachy <pizza@shaftnet.org>
|
2019-04-17 13:58:42 -04:00
|
|
|
*
|
|
|
|
* The latest version of this program can be found at:
|
|
|
|
*
|
|
|
|
* http://git.shaftnet.org/cgit/selphy_print.git
|
|
|
|
*
|
|
|
|
* 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
|
2020-01-17 16:50:56 -05:00
|
|
|
* along with this program; if not, see <https://www.gnu.org/licenses/>.
|
2019-04-17 13:58:42 -04:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: GPL-3.0+
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define BACKEND sonyupdneo_backend
|
|
|
|
|
|
|
|
#include "backend_common.h"
|
|
|
|
|
|
|
|
/* Private data structures */
|
|
|
|
struct updneo_printjob {
|
|
|
|
uint8_t *databuf;
|
|
|
|
int datalen;
|
2019-11-01 23:49:01 -04:00
|
|
|
uint8_t *hdrbuf;
|
|
|
|
int hdrlen;
|
|
|
|
uint8_t *ftrbuf;
|
|
|
|
int ftrlen;
|
2019-04-17 13:58:42 -04:00
|
|
|
|
2019-09-28 10:54:10 -04:00
|
|
|
// int copies_offset; // XXX eventually implement
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
int copies;
|
|
|
|
|
|
|
|
uint16_t rows;
|
|
|
|
uint16_t cols;
|
|
|
|
};
|
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
struct updneo_sts {
|
|
|
|
uint16_t scdiv;
|
|
|
|
uint32_t scsyv;
|
|
|
|
char scsno[17]; /* 16 char string, leading 0s */
|
|
|
|
char scsys[23]; /* 22 char string, mostly unknown */
|
|
|
|
uint16_t scmds[5];
|
|
|
|
uint16_t scprs;
|
|
|
|
uint16_t scses;
|
|
|
|
uint16_t scwts;
|
|
|
|
uint16_t scjbs;
|
|
|
|
uint8_t scsye;
|
|
|
|
uint16_t scmde;
|
|
|
|
uint8_t scmce;
|
|
|
|
char scjbi[17]; /* 16 char string, unknown */
|
|
|
|
char scsyi[31]; /* 30 char string, unknown */
|
|
|
|
uint32_t scsvi[2]; /* 2* 6char numbers */
|
|
|
|
uint32_t scmni[2]; /* 2* 6char numbers */
|
|
|
|
char sccai[15]; /* 14 char string, unknown */
|
|
|
|
uint16_t scgai;
|
|
|
|
uint8_t scgsi;
|
|
|
|
uint32_t scmdi;
|
|
|
|
};
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
struct updneo_ctx {
|
2020-08-11 20:27:26 -04:00
|
|
|
struct dyesub_connection *conn;
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
int native_bpp;
|
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
struct updneo_sts sts;
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
struct marker marker;
|
|
|
|
};
|
|
|
|
|
2019-11-02 17:01:32 -04:00
|
|
|
/* Forward declaration */
|
|
|
|
static int updneo_get_status(struct updneo_ctx *ctx);
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
/* Now for the code */
|
2019-11-06 15:14:23 -05:00
|
|
|
static const char* updneo_decode_errors(uint16_t mde, uint8_t mce, uint8_t sye)
|
|
|
|
{
|
|
|
|
if (!mde && !mce && sye)
|
|
|
|
return "None";
|
|
|
|
if (mde == 0x0800 || mce == 0x1)
|
|
|
|
return "Cover open";
|
|
|
|
if (mde == 0x0a00)
|
|
|
|
return "No paper loaded";
|
|
|
|
if (mde == 0x0002)
|
|
|
|
return "No ribbon loaded";
|
|
|
|
if (mde == 0x2000)
|
|
|
|
return "Job does not match installed media";
|
|
|
|
|
|
|
|
return "Unknown";
|
|
|
|
}
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
static void* updneo_init(void)
|
|
|
|
{
|
|
|
|
struct updneo_ctx *ctx = malloc(sizeof(struct updneo_ctx));
|
|
|
|
if (!ctx) {
|
2019-10-03 20:26:00 -04:00
|
|
|
ERROR("Memory Allocation Failure!\n");
|
2019-04-17 13:58:42 -04:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
memset(ctx, 0, sizeof(struct updneo_ctx));
|
|
|
|
return ctx;
|
|
|
|
}
|
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
static int updneo_attach(void *vctx, struct dyesub_connection *conn, uint8_t jobid)
|
2019-04-17 13:58:42 -04:00
|
|
|
{
|
|
|
|
struct updneo_ctx *ctx = vctx;
|
2019-11-02 17:01:32 -04:00
|
|
|
int ret;
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
UNUSED(jobid);
|
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
ctx->conn = conn;
|
2019-04-17 13:58:42 -04:00
|
|
|
|
2019-11-03 07:03:57 -05:00
|
|
|
if (test_mode < TEST_MODE_NOATTACH) {
|
|
|
|
if ((ret = updneo_get_status(ctx))) {
|
|
|
|
return ret;
|
|
|
|
}
|
2019-11-02 17:01:32 -04:00
|
|
|
}
|
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPD898) {
|
2019-04-17 13:58:42 -04:00
|
|
|
ctx->marker.color = "#000000"; /* Ie black! */
|
|
|
|
ctx->native_bpp = 1;
|
2019-11-06 15:14:23 -05:00
|
|
|
|
|
|
|
ctx->marker.name = "Unknown";
|
|
|
|
ctx->marker.numtype = -1;
|
2019-12-11 23:18:39 -05:00
|
|
|
ctx->marker.levelmax = CUPS_MARKER_UNAVAILABLE;
|
|
|
|
ctx->marker.levelnow = CUPS_MARKER_UNKNOWN;
|
2019-04-17 13:58:42 -04:00
|
|
|
} else {
|
|
|
|
ctx->marker.color = "#00FFFF#FF00FF#FFFF00";
|
|
|
|
ctx->native_bpp = 3;
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->marker.levelmax = 50;
|
|
|
|
ctx->marker.numtype = (ctx->sts.scmdi >> 16) & 0xff;
|
|
|
|
ctx->marker.levelnow = ctx->sts.scmds[4];
|
|
|
|
|
|
|
|
if (test_mode >= TEST_MODE_NOATTACH && getenv("MEDIA_CODE"))
|
|
|
|
ctx->marker.numtype = atoi(getenv("MEDIA_CODE"));
|
2019-04-17 13:58:42 -04:00
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
if (ctx->marker.numtype == 0x11) { /* UP-DR80MD */
|
|
|
|
ctx->marker.name = "UPC-R81MD (Letter)"; // vs UPC-R80MD (A4)
|
|
|
|
} else {
|
|
|
|
ctx->marker.name = "Unknown";
|
|
|
|
}
|
|
|
|
}
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void updneo_cleanup_job(const void *vjob)
|
|
|
|
{
|
|
|
|
const struct updneo_printjob *job = vjob;
|
|
|
|
|
|
|
|
if (job->databuf)
|
|
|
|
free(job->databuf);
|
2019-11-01 23:49:01 -04:00
|
|
|
if (job->hdrbuf)
|
|
|
|
free(job->hdrbuf);
|
|
|
|
if (job->ftrbuf)
|
|
|
|
free(job->ftrbuf);
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
free((void*)job);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define MAX_PRINTJOB_LEN (3400*2392*3 + 2048)
|
|
|
|
|
|
|
|
static int updneo_read_parse(void *vctx, const void **vjob, int data_fd, int copies) {
|
|
|
|
struct updneo_ctx *ctx = vctx;
|
2019-11-01 23:49:01 -04:00
|
|
|
int run = 1;
|
|
|
|
|
|
|
|
uint8_t tmpbuf[257];
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
struct updneo_printjob *job = NULL;
|
|
|
|
|
|
|
|
if (!ctx)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
2019-11-01 23:49:01 -04:00
|
|
|
/* Allocate job */
|
2019-04-17 13:58:42 -04:00
|
|
|
job = malloc(sizeof(*job));
|
|
|
|
if (!job) {
|
|
|
|
ERROR("Memory allocation failure!\n");
|
|
|
|
return CUPS_BACKEND_RETRY_CURRENT;
|
|
|
|
}
|
|
|
|
memset(job, 0, sizeof(*job));
|
|
|
|
|
|
|
|
/* Read in data chunks. */
|
|
|
|
while(run) {
|
2019-11-01 23:49:01 -04:00
|
|
|
uint8_t *ptr = NULL;
|
|
|
|
int i, len, *lenptr;
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
/* Read in data block header (256 bytes) */
|
2019-11-01 23:49:01 -04:00
|
|
|
i = read(data_fd, tmpbuf, 256);
|
2019-04-17 13:58:42 -04:00
|
|
|
if (i < 0) {
|
2019-11-14 17:02:11 -05:00
|
|
|
ERROR("Read failed (%d)\n", i);
|
2019-04-17 13:58:42 -04:00
|
|
|
updneo_cleanup_job(job);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
if (i == 0)
|
|
|
|
break;
|
|
|
|
|
2019-11-01 23:49:01 -04:00
|
|
|
/* Explicitly null terminate just in case */
|
|
|
|
tmpbuf[256] = 0;
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
/* Parse header. Format:
|
|
|
|
|
|
|
|
JOBSIZE=pdlname,blocklen,printsize,arg1,..,argN<NULL>
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2019-11-01 23:49:01 -04:00
|
|
|
if (strncmp("JOBSIZE=", (char*) tmpbuf, 8)) {
|
2019-05-06 21:10:43 -04:00
|
|
|
updneo_cleanup_job(job);
|
2019-04-17 13:58:42 -04:00
|
|
|
ERROR("Invalid spool format!\n");
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
|
2019-11-01 23:49:01 -04:00
|
|
|
/* PDL type */
|
2019-11-02 14:32:12 -04:00
|
|
|
char *tok = strtok((char*)&tmpbuf[8], "\r\n,");
|
2019-04-17 13:58:42 -04:00
|
|
|
if (!tok) {
|
2019-05-06 21:10:43 -04:00
|
|
|
updneo_cleanup_job(job);
|
2019-04-17 13:58:42 -04:00
|
|
|
ERROR("Invalid spool format (PDL)!\n");
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Payload length */
|
2019-11-01 23:49:01 -04:00
|
|
|
char *tokl = strtok(NULL, "\r\n,");
|
|
|
|
if (!tokl) {
|
2019-05-12 08:17:56 -04:00
|
|
|
updneo_cleanup_job(job);
|
2019-04-17 13:58:42 -04:00
|
|
|
ERROR("Invalid spool format (block length missing)!\n");
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
2019-11-01 23:49:01 -04:00
|
|
|
len = atoi(tokl);
|
2019-04-17 13:58:42 -04:00
|
|
|
if (len == 0 || len > MAX_PRINTJOB_LEN) {
|
2019-05-12 08:17:56 -04:00
|
|
|
updneo_cleanup_job(job);
|
2019-04-17 13:58:42 -04:00
|
|
|
ERROR("Invalid spool format (block length %d)!\n", len);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
2019-11-01 23:49:01 -04:00
|
|
|
|
|
|
|
/* Behavior based on the various PDL blocks */
|
|
|
|
if (!strncmp("PJL-H", tok, 5)) {
|
|
|
|
job->hdrbuf = malloc(len);
|
|
|
|
if (!job->hdrbuf) {
|
|
|
|
ERROR("Memory allocation failure!\n");
|
|
|
|
updneo_cleanup_job(job);
|
|
|
|
return CUPS_BACKEND_RETRY_CURRENT;
|
|
|
|
}
|
|
|
|
ptr = job->hdrbuf;
|
|
|
|
lenptr = &job->hdrlen;
|
|
|
|
} else if (!strncmp("PJL-T", tok, 5)) {
|
|
|
|
job->ftrbuf = malloc(len);
|
|
|
|
if (!job->ftrbuf) {
|
|
|
|
ERROR("Memory allocation failure!\n");
|
|
|
|
updneo_cleanup_job(job);
|
|
|
|
return CUPS_BACKEND_RETRY_CURRENT;
|
|
|
|
}
|
|
|
|
ptr = job->ftrbuf;
|
|
|
|
lenptr = &job->ftrlen;
|
|
|
|
run = 0;
|
2019-11-02 14:32:12 -04:00
|
|
|
} else if (!strncmp("PDL", tok, 3)) {
|
2019-11-01 23:49:01 -04:00
|
|
|
job->databuf = malloc(len);
|
|
|
|
if (!job->databuf) {
|
|
|
|
ERROR("Memory allocation failure!\n");
|
|
|
|
updneo_cleanup_job(job);
|
|
|
|
return CUPS_BACKEND_RETRY_CURRENT;
|
|
|
|
}
|
|
|
|
ptr = job->databuf;
|
|
|
|
lenptr = &job->datalen;
|
|
|
|
} else {
|
|
|
|
ERROR("Unrecognized PDL type '%s'\n", tok);
|
|
|
|
updneo_cleanup_job(job);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// DEBUG("Read block '%s' @ %d ...\n", tok, job->datalen);
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
// DEBUG("...len '%d'\n", len);
|
|
|
|
|
|
|
|
// parse the rest?
|
|
|
|
// 898MD: 6,0,0,0
|
|
|
|
// D80MD: 4
|
|
|
|
// CR20L: 64,0,0,0
|
|
|
|
|
|
|
|
/* Read in the data chunk */
|
|
|
|
while(len > 0) {
|
2019-11-01 23:49:01 -04:00
|
|
|
i = read(data_fd, ptr + *lenptr, len);
|
2019-04-17 13:58:42 -04:00
|
|
|
if (i < 0) {
|
|
|
|
updneo_cleanup_job(job);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
if (i == 0)
|
|
|
|
break;
|
|
|
|
|
2019-11-01 23:49:01 -04:00
|
|
|
*lenptr += i;
|
2019-04-17 13:58:42 -04:00
|
|
|
len -= i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-01 23:49:01 -04:00
|
|
|
if (!job->datalen || !job->hdrlen || !job->ftrlen) {
|
2019-11-14 17:02:11 -05:00
|
|
|
if (job->datalen + job->hdrlen + job->ftrlen) {
|
|
|
|
ERROR("Necessary block missing!\n");
|
|
|
|
}
|
2019-04-17 13:58:42 -04:00
|
|
|
updneo_cleanup_job(job);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sanity check job parameters */
|
2019-11-06 15:14:23 -05:00
|
|
|
// XXX rows * cols lines up with imgsize, and others?
|
|
|
|
// Check vs loaded media type (ctx->marker.numtype, plus scsyi)
|
2019-04-17 13:58:42 -04:00
|
|
|
|
2019-11-01 23:49:01 -04:00
|
|
|
// set job copies to max(job, parameter)
|
|
|
|
// job->copies = copies;
|
|
|
|
job->copies = 1; /* Printer makes copies */
|
|
|
|
UNUSED(copies);
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
*vjob = job;
|
|
|
|
|
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
2019-11-02 08:26:15 -04:00
|
|
|
static int dlen;
|
|
|
|
static struct deviceid_dict dict[MAX_DICT];
|
|
|
|
|
|
|
|
static int updneo_get_status(struct updneo_ctx *ctx)
|
|
|
|
{
|
2020-08-11 20:27:26 -04:00
|
|
|
char *ieee_id = get_device_id(ctx->conn->dev, ctx->conn->iface);
|
2019-11-06 15:14:23 -05:00
|
|
|
int i;
|
2019-11-02 08:26:15 -04:00
|
|
|
|
|
|
|
if (!ieee_id)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
2019-11-05 16:43:52 -05:00
|
|
|
/* Don't forget to log! */
|
|
|
|
if (dyesub_debug >= 1) {
|
2019-11-05 17:40:12 -05:00
|
|
|
DEBUG("IEEE1284: %s\n", ieee_id);
|
2019-11-05 16:43:52 -05:00
|
|
|
}
|
|
|
|
|
2019-11-02 08:26:15 -04:00
|
|
|
dlen = parse1284_data(ieee_id, dict);
|
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
/* Parse out data */
|
|
|
|
for (i = 0; i < dlen ; i++) {
|
2019-11-13 19:59:49 -05:00
|
|
|
if (!strcmp("SCDIV", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scdiv = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCSYV", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scsyv = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCSNO", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
strncpy(ctx->sts.scsno, dict[i].val, sizeof(ctx->sts.scsno) - 1);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCSYS", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
strncpy(ctx->sts.scsys, dict[i].val, sizeof(ctx->sts.scsys) - 1);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCMDS", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
int j;
|
|
|
|
char buf[5];
|
|
|
|
buf[4] = 0;
|
|
|
|
for (j = 0 ; j < 5 ; j++) {
|
|
|
|
memcpy(buf, dict[i].val + (4*j), 4);
|
|
|
|
ctx->sts.scmds[j] = strtol(buf, NULL, 16);
|
|
|
|
}
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCPRS", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scprs = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCSES", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scses = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCWTS", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scwts = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCJBS", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scjbs = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCSYE", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scsye = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCMDE", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scmde = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCMCE", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scmce = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCJBI", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
strncpy(ctx->sts.scjbi, dict[i].val, sizeof(ctx->sts.scjbi) - 1);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCSYI", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
strncpy(ctx->sts.scsyi, dict[i].val, sizeof(ctx->sts.scsyi) - 1);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCSVI", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
int j;
|
|
|
|
char buf[7];
|
|
|
|
buf[6] = 0;
|
2019-11-07 12:34:21 -05:00
|
|
|
for (j = 0 ; j < 2 ; j++) {
|
2019-11-06 15:14:23 -05:00
|
|
|
memcpy(buf, dict[i].val + (6*j), 6);
|
|
|
|
ctx->sts.scsvi[j] = strtol(buf, NULL, 16);
|
|
|
|
}
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCMNI", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
int j;
|
|
|
|
char buf[7];
|
|
|
|
buf[6] = 0;
|
2019-11-07 12:34:21 -05:00
|
|
|
for (j = 0 ; j < 2 ; j++) {
|
2019-11-06 15:14:23 -05:00
|
|
|
memcpy(buf, dict[i].val + (6*j), 6);
|
|
|
|
ctx->sts.scmni[j] = strtol(buf, NULL, 16);
|
|
|
|
}
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCCAI", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
strncpy(ctx->sts.sccai, dict[i].val, sizeof(ctx->sts.sccai) - 1);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCGAI", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scgai = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCGSI", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scgsi = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("SCMDI", dict[i].key)) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->sts.scmdi = strtol(dict[i].val, NULL, 16);
|
2019-11-13 19:59:49 -05:00
|
|
|
} else if (!strcmp("MFG", dict[i].key) ||
|
|
|
|
!strcmp("MDL", dict[i].key) ||
|
|
|
|
!strcmp("DES", dict[i].key) ||
|
|
|
|
!strcmp("CMD", dict[i].key) ||
|
|
|
|
!strcmp("CLS", dict[i].key))
|
2019-11-06 15:14:23 -05:00
|
|
|
{
|
|
|
|
/* Ignore standard IEEE1284 attributes! */
|
|
|
|
} else {
|
2019-11-15 20:48:13 -05:00
|
|
|
if (!strncmp("SC", dict[i].key, 2))
|
|
|
|
DEBUG("Extra/Unknown IEEE1284 field '%s' = '%s'\n",
|
|
|
|
dict[i].key, dict[i].val);
|
2019-11-06 15:14:23 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
};
|
2019-11-02 08:26:15 -04:00
|
|
|
|
|
|
|
/* Clean up */
|
|
|
|
if (ieee_id) free(ieee_id);
|
|
|
|
while (dlen--) {
|
|
|
|
free (dict[dlen].key);
|
|
|
|
free (dict[dlen].val);
|
|
|
|
}
|
2019-11-02 17:01:32 -04:00
|
|
|
|
|
|
|
return CUPS_BACKEND_OK;
|
2019-11-02 08:26:15 -04:00
|
|
|
}
|
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
static void updneo_dump_status(struct updneo_sts *sts)
|
|
|
|
{
|
|
|
|
/* Dump status */
|
|
|
|
INFO("Serial Number: %s\n", sts->scsno);
|
2019-11-14 15:29:56 -05:00
|
|
|
INFO("Firmware Version: %02x.%02x.%02x.%02x\n",
|
2019-11-06 15:14:23 -05:00
|
|
|
(sts->scsyv >> 24) & 0xff,
|
|
|
|
(sts->scsyv >> 16) & 0xff,
|
|
|
|
(sts->scsyv >> 8) & 0xff,
|
|
|
|
(sts->scsyv >> 0) & 0xff);
|
|
|
|
INFO("Media type: %s\n", sts->scmdi == 0x110154 ? "UPC-R81MD (Letter)" : "Unknown");
|
|
|
|
INFO("Remaining prints: %u/50\n", sts->scmds[4]);
|
|
|
|
INFO("Print count: %u\n", sts->scsvi[0]);
|
|
|
|
|
|
|
|
/* If the printer reports an error, pass it on */
|
|
|
|
if (sts->scmde || sts->scmce || sts->scsye) {
|
|
|
|
ERROR("Printer error: %s (MD=%04x, MC=%02x, SY=%02x)\n",
|
|
|
|
updneo_decode_errors(sts->scmde, sts->scmce, sts->scsye),
|
|
|
|
sts->scmde, sts->scmce, sts->scsye);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
static int updneo_main_loop(void *vctx, const void *vjob) {
|
|
|
|
struct updneo_ctx *ctx = vctx;
|
|
|
|
int ret;
|
|
|
|
int copies;
|
|
|
|
|
|
|
|
const struct updneo_printjob *job = vjob;
|
|
|
|
|
|
|
|
if (!ctx)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
if (!job)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
|
|
|
copies = job->copies;
|
|
|
|
|
|
|
|
top:
|
|
|
|
|
2019-11-02 17:01:32 -04:00
|
|
|
/* Query printer status */
|
|
|
|
if ((ret = updneo_get_status(ctx))) {
|
|
|
|
return ret;
|
|
|
|
}
|
2019-04-17 13:58:42 -04:00
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
/* If the printer reports an error, bail */
|
|
|
|
if (ctx->sts.scmde || ctx->sts.scmce || ctx->sts.scsye) {
|
|
|
|
ERROR("Printer error: %s (MD=%04x, MC=%02x, SY=%02x)\n",
|
|
|
|
updneo_decode_errors(ctx->sts.scmde, ctx->sts.scmce, ctx->sts.scsye),
|
|
|
|
ctx->sts.scmde, ctx->sts.scmce, ctx->sts.scsye);
|
|
|
|
return CUPS_BACKEND_STOP;
|
|
|
|
}
|
|
|
|
/* Wait for the printer to become idle */
|
|
|
|
if (ctx->sts.scprs) {
|
|
|
|
sleep(1);
|
|
|
|
goto top;
|
|
|
|
}
|
2019-11-01 23:49:01 -04:00
|
|
|
|
|
|
|
/* Send over header */
|
2020-08-11 20:27:26 -04:00
|
|
|
if ((ret = send_data(ctx->conn,
|
2019-11-01 23:49:01 -04:00
|
|
|
job->hdrbuf, job->hdrlen)))
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
|
|
|
/* Send over data */
|
2020-08-11 20:27:26 -04:00
|
|
|
if ((ret = send_data(ctx->conn,
|
2019-04-17 13:58:42 -04:00
|
|
|
job->databuf, job->datalen)))
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
2019-11-01 23:49:01 -04:00
|
|
|
/* Send over footer */
|
2020-08-11 20:27:26 -04:00
|
|
|
if ((ret = send_data(ctx->conn,
|
2019-11-01 23:49:01 -04:00
|
|
|
job->ftrbuf, job->ftrlen)))
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
/* Wait for completion! */
|
|
|
|
retry:
|
|
|
|
sleep(1);
|
|
|
|
|
2019-11-02 17:01:32 -04:00
|
|
|
if ((ret = updneo_get_status(ctx))) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
/* If the printer reports an error, bail */
|
|
|
|
if (ctx->sts.scmde || ctx->sts.scmce || ctx->sts.scsye) {
|
|
|
|
ERROR("Printer error: %s (MD=%04x, MC=%02x, SY=%02x)\n",
|
|
|
|
updneo_decode_errors(ctx->sts.scmde, ctx->sts.scmce, ctx->sts.scsye),
|
|
|
|
ctx->sts.scmde, ctx->sts.scmce, ctx->sts.scsye);
|
|
|
|
return CUPS_BACKEND_STOP;
|
|
|
|
}
|
|
|
|
|
2019-11-14 15:29:56 -05:00
|
|
|
/* See if we're busy... */
|
|
|
|
if (ctx->sts.scprs != 0) {
|
|
|
|
if (fast_return) {
|
|
|
|
INFO("Fast return mode enabled.\n");
|
|
|
|
} else {
|
|
|
|
goto retry;
|
|
|
|
}
|
2019-04-17 13:58:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Clean up */
|
|
|
|
if (terminate)
|
|
|
|
copies = 1;
|
|
|
|
|
|
|
|
INFO("Print complete (%d copies remaining)\n", copies - 1);
|
|
|
|
|
|
|
|
if (copies && --copies) {
|
|
|
|
goto top;
|
|
|
|
}
|
|
|
|
|
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
static void updneo_cmdline(void)
|
|
|
|
{
|
|
|
|
DEBUG("\t\t[ -s ] # Query status\n");
|
|
|
|
}
|
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
static int updneo_cmdline_arg(void *vctx, int argc, char **argv)
|
|
|
|
{
|
|
|
|
struct updneo_ctx *ctx = vctx;
|
|
|
|
int i, j = 0;
|
|
|
|
|
|
|
|
if (!ctx)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
while ((i = getopt(argc, argv, GETOPT_LIST_GLOBAL "s")) >= 0) {
|
|
|
|
switch(i) {
|
|
|
|
GETOPT_PROCESS_GLOBAL
|
2019-11-02 17:01:32 -04:00
|
|
|
case 's':
|
|
|
|
j = updneo_get_status(ctx);
|
2019-11-06 15:14:23 -05:00
|
|
|
if (!j)
|
|
|
|
updneo_dump_status(&ctx->sts);
|
2019-11-02 17:01:32 -04:00
|
|
|
break;
|
2019-04-17 13:58:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (j) return j;
|
|
|
|
}
|
|
|
|
|
2019-11-06 15:14:23 -05:00
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
static int updneo_query_serno(struct dyesub_connection *conn, char *buf, int buf_len)
|
2019-11-06 15:14:23 -05:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
char *ptr;
|
|
|
|
struct updneo_ctx ctx = {
|
2020-08-11 20:27:26 -04:00
|
|
|
.conn = conn,
|
2019-11-06 15:14:23 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
if ((ret = updneo_get_status(&ctx))) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
ptr = ctx.sts.scsno;
|
2019-11-17 20:54:08 -05:00
|
|
|
while (*ptr == 0x30) ptr++;
|
2019-11-06 15:14:23 -05:00
|
|
|
strncpy(buf, ptr, buf_len);
|
|
|
|
buf[buf_len-1] = 0;
|
|
|
|
|
|
|
|
return CUPS_BACKEND_OK;
|
2019-04-17 13:58:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static int updneo_query_markers(void *vctx, struct marker **markers, int *count)
|
|
|
|
{
|
|
|
|
struct updneo_ctx *ctx = vctx;
|
2019-11-02 17:01:32 -04:00
|
|
|
int ret;
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
*markers = &ctx->marker;
|
|
|
|
*count = 1;
|
|
|
|
|
2019-11-02 17:01:32 -04:00
|
|
|
/* Query printer status */
|
|
|
|
if ((ret = updneo_get_status(ctx))) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type != P_SONY_UPD898) {
|
2019-11-06 15:14:23 -05:00
|
|
|
ctx->marker.levelnow = ctx->sts.scmds[4];
|
|
|
|
}
|
2019-04-17 13:58:42 -04:00
|
|
|
|
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *sonyupdneo_prefixes[] = {
|
2020-01-30 22:08:55 -05:00
|
|
|
"sonyupdneo", /* Family Name */
|
|
|
|
"dnp-sl20", // extra, unknown if shared with CR20L
|
2019-04-17 13:58:42 -04:00
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Exported */
|
|
|
|
#define USB_VID_SONY 0x054C
|
|
|
|
#define USB_PID_SONY_UPD898MD 0xabcd // 0x589a?
|
|
|
|
#define USB_PID_SONY_UPCR20L 0xbcde
|
2019-11-01 22:12:07 -04:00
|
|
|
#define USB_PID_SONY_UPDR80MD 0x03c3
|
2020-02-27 07:12:42 -05:00
|
|
|
#define USB_PID_STRYKER_SDP1000 0x03c4
|
2019-11-01 22:12:07 -04:00
|
|
|
#define USB_PID_SONY_UPDR80 0x03c5
|
2019-05-24 17:29:22 -04:00
|
|
|
#define USB_PID_SONY_UPCX1 0x02d4
|
2019-04-17 13:58:42 -04:00
|
|
|
|
2020-10-05 22:26:39 -04:00
|
|
|
const struct dyesub_backend sonyupdneo_backend = {
|
2019-04-17 13:58:42 -04:00
|
|
|
.name = "Sony UP-D Neo",
|
2020-08-11 20:27:26 -04:00
|
|
|
.version = "0.11",
|
2019-04-17 13:58:42 -04:00
|
|
|
.uri_prefixes = sonyupdneo_prefixes,
|
|
|
|
.cmdline_arg = updneo_cmdline_arg,
|
2019-11-06 15:14:23 -05:00
|
|
|
.cmdline_usage = updneo_cmdline,
|
2019-04-17 13:58:42 -04:00
|
|
|
.init = updneo_init,
|
|
|
|
.attach = updneo_attach,
|
|
|
|
.cleanup_job = updneo_cleanup_job,
|
|
|
|
.read_parse = updneo_read_parse,
|
|
|
|
.main_loop = updneo_main_loop,
|
|
|
|
.query_markers = updneo_query_markers,
|
2019-11-06 15:14:23 -05:00
|
|
|
.query_serno = updneo_query_serno,
|
2019-04-17 13:58:42 -04:00
|
|
|
.devices = {
|
|
|
|
{ USB_VID_SONY, USB_PID_SONY_UPD898MD, P_SONY_UPD898, NULL, "sony-upd898"},
|
|
|
|
{ USB_VID_SONY, USB_PID_SONY_UPCR20L, P_SONY_UPCR20L, NULL, "sony-upcr20l"},
|
2019-11-01 22:12:07 -04:00
|
|
|
{ USB_VID_SONY, USB_PID_SONY_UPDR80, P_SONY_UPDR80, NULL, "sony-updr80"},
|
2019-04-20 19:55:36 -04:00
|
|
|
{ USB_VID_SONY, USB_PID_SONY_UPDR80MD, P_SONY_UPDR80, NULL, "sony-updr80md"},
|
2020-02-27 07:12:42 -05:00
|
|
|
{ USB_VID_SONY, USB_PID_STRYKER_SDP1000, P_SONY_UPDR80, NULL, "stryker-sdp1000"},
|
2019-11-01 22:12:07 -04:00
|
|
|
|
2019-04-17 13:58:42 -04:00
|
|
|
{ 0, 0, 0, NULL, NULL}
|
|
|
|
}
|
|
|
|
};
|
2019-05-05 19:33:16 -04:00
|
|
|
|
|
|
|
/* Sony UP-D (new) printer spool format
|
|
|
|
|
|
|
|
Covers UP-CR20L, UP-DR80/DR80MD, UP-D898/UP-X898
|
|
|
|
|
2019-11-01 22:12:07 -04:00
|
|
|
HP-PJL wrapper around custom Sony PDL:
|
|
|
|
|
|
|
|
JOBSIZE=PJL-H,size,arg1,arg2,etc [null terminated, padded to 256 bytes]
|
|
|
|
[ size bytes of PJL header! ]
|
|
|
|
JOBSIZE=PDL,size,args [null terminated, padded to 256 bytes]
|
|
|
|
[ size bytes of PDL data! ]
|
|
|
|
JOBSIZE=PJL-T,size,args [null terminated, padded to 256 bytes]
|
|
|
|
[ size bytes of PJL trailer! ]
|
|
|
|
|
|
|
|
PJL header:
|
|
|
|
|
|
|
|
<ESC>%-12345X<CR><LF>
|
|
|
|
@PJL COMMENT free form text here <CR><LF>
|
|
|
|
@PJL JOB NAME="name me" ID="someid"<CR><LF>
|
|
|
|
@PJL .... <CR><LF>
|
|
|
|
@PJL ENTER LANGUAGE=SONY-PDL-DS2<CR><LF>
|
|
|
|
|
|
|
|
PJL footer:
|
2019-05-05 19:33:16 -04:00
|
|
|
|
2019-11-01 22:12:07 -04:00
|
|
|
@PJL EOJ<CR><LF>
|
|
|
|
<ESC>%-12345X<CR><LF>
|
2019-05-05 19:33:16 -04:00
|
|
|
|
2019-11-01 22:32:56 -04:00
|
|
|
PDL notes:
|
2019-05-05 19:33:16 -04:00
|
|
|
|
2019-11-01 22:32:56 -04:00
|
|
|
size is the length mentioned in the payload (ie rows * cols * planes)
|
|
|
|
plus the PDL header (varies) and PDL footer (7 bytes)
|
2019-05-05 19:33:16 -04:00
|
|
|
|
2019-11-01 22:32:56 -04:00
|
|
|
|
|
|
|
UP-D898MD: 18*16+2 == 290 byte header
|
2019-05-05 19:33:16 -04:00
|
|
|
|
|
|
|
00000250 00 00 YY YY = rows
|
|
|
|
00000260 01 00 00 10 0f 00 1c 00 00 00 00 00 00 00 00 00 XX XX = columns (fixed at 05 00)
|
|
|
|
00000270 00 00 00 00 00 01 02 00 09 00 NN 01 00 11 01 08 NN = Copies (01..?) <-- GUESS
|
|
|
|
00000280 00 1a 00 00 00 00 XX XX YY YY 09 00 28 01 00 d4
|
|
|
|
00000290 00 00 03 58 YY YY 00 00 13 01 00 04 00 80 00 23
|
|
|
|
000002a0 00 0c 01 09 XX XX YY YY 00 00 00 00 08 ff 08 00
|
|
|
|
000002b0 19 00 00 00 00 XX XX YY YY 00 00 81 80 00 8f 00
|
|
|
|
000002c0 b8 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
000002d0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
000002f0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000300 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000310 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000320 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000330 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000340 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000350 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000360 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000370 00 00 00 00 00 00 00 00 00 c0 00 82 LL LL LL LL LL == payload bytes, BE (== XX * YY * 1)
|
|
|
|
|
|
|
|
[payload of LL bytes follows]
|
|
|
|
|
2019-11-01 22:32:56 -04:00
|
|
|
UP-CR20L: 290 byte header, 330 dpi, 1210x1728/1382x2048/1728*2380/2724*2048 (L/PC/2L/2PC)
|
2019-05-05 19:33:16 -04:00
|
|
|
|
|
|
|
00000250 00 00
|
|
|
|
00000260 01 00 00 10 0f 00 1c 00 00 00 00 00 00 00 00 00
|
|
|
|
00000270 00 00 01 00 00 00 02 00 16 00 00 02 00 09 00 NN NN = Copies (01..??)
|
|
|
|
00000280 02 00 06 01 01 03 00 1d 00 00 00 01 00 20 01 01
|
|
|
|
00000290 00 27 40 01 00 11 01 08 00 1a 00 00 00 00 RR RR
|
|
|
|
000002a0 CC CC 00 00 13 01 00 04 00 80 00 23 00 10 03 00 CC CC == Columns (BE)
|
|
|
|
000002b0 RR RR CC CC 00 00 00 00 08 08 08 ff ff ff 01 00 RR RR == Rows (BE)
|
|
|
|
000002c0 17 00 08 00 19 00 00 00 00 RR RR CC CC 00 00 81
|
|
|
|
000002d0 80 00 8f 00 a4 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
000002e0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
000002f0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000300 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000310 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000320 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000330 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000340 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000350 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000360 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000370 00 00 00 00 00 00 00 00 00 c0 00 82 LL LL LL LL LL == payload bytes, BE (== RR * CC * 3)
|
|
|
|
|
|
|
|
[payload of LL bytes follows]
|
|
|
|
|
2019-11-01 22:32:56 -04:00
|
|
|
UP-DR80MD: (19*16+8) = 312 byte header (A4 = 3400x2392, Letter = 3192*2464)
|
2019-05-05 19:33:16 -04:00
|
|
|
|
|
|
|
00000240 00 00 01 00 00 10 0f 00
|
|
|
|
00000250 1c 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ZZ ZZ = 0x00 (Letter) 0x56 (A4)
|
|
|
|
00000260 02 00 16 00 01 80 00 15 00 12 55 50 44 52 38 30 SS = 0x00 (LUT0) 0xff (No LUT)
|
|
|
|
00000270 00 00 4c 55 54 QQ 00 00 00 00 00 SS 02 00 09 00 QQ = 0x30 (LUT0) 0x2f (No LUT)
|
|
|
|
00000280 NN 02 00 06 01 03 04 00 1d 01 00 00 05 01 00 20 NN = Copies (01...??)
|
|
|
|
00000290 00 01 00 11 01 08 00 1a 00 00 00 00 CC CC RR RR RR RR = Rows (BE)
|
|
|
|
000002a0 00 00 13 01 00 04 00 80 00 23 00 10 03 00 CC CC CC CC = Cols (BE)
|
|
|
|
000002b0 RR RR 00 00 00 00 08 08 08 ff ff ff 01 00 17 00
|
|
|
|
000002c0 08 00 19 00 00 00 00 CC CC RR RR 00 00 81 80 00
|
|
|
|
000002d0 8f 00 a6 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
000002e0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
000002f0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000300 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000310 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000320 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000330 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000340 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000350 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000360 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
|
|
|
|
00000370 00 00 00 00 00 00 00 00 00 c0 00 82 LL LL LL LL LL == payload, BE (== RR * CC * 3)
|
|
|
|
|
|
|
|
[payload of LL bytes follows]
|
|
|
|
|
|
|
|
Common sequences:
|
|
|
|
|
|
|
|
[ D898 ]
|
|
|
|
|
|
|
|
[00 00 01 00 00 10 0f 00 1c 00 00 00 00 00 00 00
|
|
|
|
00 00 00 00]00 00 00 01 [02 00 09 00 NN]01 00 11
|
|
|
|
01 08 00 1a 00 00 00 00 XX XX |