2013-07-14 15:55:35 -04:00
|
|
|
/*
|
2019-03-23 15:08:58 -04:00
|
|
|
* Sony UP-D series Photo Printer CUPS backend -- libusb-1.0 version
|
2013-07-14 15:55:35 -04:00
|
|
|
*
|
2020-06-30 23:42:54 -04:00
|
|
|
* (c) 2013-2020 Solomon Peachy <pizza@shaftnet.org>
|
2013-07-14 15:55:35 -04:00
|
|
|
*
|
|
|
|
* The latest version of this program can be found at:
|
2014-01-13 05:41:48 -05:00
|
|
|
*
|
2013-08-20 20:10:21 -04:00
|
|
|
* http://git.shaftnet.org/cgit/selphy_print.git
|
2014-01-13 05:41:48 -05:00
|
|
|
*
|
2013-07-14 15:55:35 -04: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
|
2020-01-17 16:50:56 -05:00
|
|
|
* along with this program; if not, see <https://www.gnu.org/licenses/>.
|
2013-07-14 15:55:35 -04:00
|
|
|
*
|
2017-11-17 13:34:26 -05:00
|
|
|
* SPDX-License-Identifier: GPL-3.0+
|
|
|
|
*
|
2013-07-14 15:55:35 -04:00
|
|
|
*/
|
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|
|
#define BACKEND sonyupd_backend
|
2015-08-13 21:09:56 -04:00
|
|
|
|
2013-07-18 08:46:44 -04:00
|
|
|
#include "backend_common.h"
|
2013-07-14 15:55:35 -04:00
|
|
|
|
2019-03-15 22:57:13 -04:00
|
|
|
/* Printer status
|
|
|
|
--> 1b e0 00 00 00 00 XX 00 [[ XX is 0xe on UPD895, 0xf on others ]]
|
|
|
|
<-- this struct
|
|
|
|
*/
|
|
|
|
struct sony_updsts {
|
2019-04-06 17:38:28 -04:00
|
|
|
uint8_t len; /* 0x0d/0x0e (ie number of bytes AFTER this one) */
|
|
|
|
uint8_t zero1; /* 0x00 */
|
2019-04-16 14:26:51 -04:00
|
|
|
uint8_t printing; /* UPD_PRINTING_* */
|
2019-03-15 22:57:13 -04:00
|
|
|
uint8_t remain; /* Number of remaining pages */
|
2020-11-28 10:39:28 -05:00
|
|
|
uint8_t sts0; /* UPD_STS0_* */
|
2019-04-16 14:26:51 -04:00
|
|
|
uint8_t sts1; /* UPD_STS1_* */
|
2020-11-28 10:39:28 -05:00
|
|
|
uint8_t sts2; /* UPD_STS2_* */
|
|
|
|
uint8_t sts3; /* UPD_STS3_* */
|
2020-11-28 09:35:24 -05:00
|
|
|
uint8_t ribbon; /* 0x04 = R206/6x8 */
|
|
|
|
uint8_t paper; /* 0x38 = EMPTY, 0xa8 = loaded */
|
2019-03-15 22:57:13 -04:00
|
|
|
uint16_t max_cols; /* BE */
|
|
|
|
uint16_t max_rows; /* BE */
|
2019-04-16 14:26:51 -04:00
|
|
|
uint8_t percent; /* 0-99, if job is printing (UP-D89x) */
|
2019-03-15 22:57:13 -04:00
|
|
|
} __attribute__((packed));
|
|
|
|
|
2020-12-01 22:47:40 -05:00
|
|
|
struct sony_prints {
|
|
|
|
uint8_t zero[4];
|
|
|
|
uint16_t remain; /* BE, remaining prints on media */
|
|
|
|
} __attribute((packed));
|
|
|
|
|
2019-04-16 14:26:51 -04:00
|
|
|
#define UPD_PRINTING_BW 0xe0 /* UPD-895/897 only */
|
|
|
|
#define UPD_PRINTING_Y 0x40
|
|
|
|
#define UPD_PRINTING_M 0x80
|
|
|
|
#define UPD_PRINTING_C 0xc0
|
|
|
|
#define UPD_PRINTING_O 0x20
|
|
|
|
#define UPD_PRINTING_IDLE 0x00
|
|
|
|
|
2020-11-28 10:39:28 -05:00
|
|
|
/* Confirmed on UP-DR200 */
|
|
|
|
#define UPD_STS0_OK 0x00
|
|
|
|
#define UPD_STS0_NORIBBON 0x10
|
|
|
|
#define UPD_STS0_NOPAPER 0x20
|
|
|
|
#define UPD_STS0_DOOROPEN 0x40
|
|
|
|
|
2019-04-16 14:26:51 -04:00
|
|
|
#define UPD_STS1_IDLE 0x00
|
|
|
|
#define UPD_STS1_DOOROPEN 0x08
|
|
|
|
#define UPD_STS1_NOPAPER 0x40
|
|
|
|
#define UPD_STS1_PRINTING 0x80
|
2020-11-29 09:42:05 -05:00
|
|
|
#define UPD_STS1_PRINTING2 0xC0
|
2019-04-16 14:26:51 -04:00
|
|
|
|
2020-11-28 09:35:24 -05:00
|
|
|
#define UPD_RIBBON_R206 0x04
|
|
|
|
|
2019-03-17 13:15:27 -04:00
|
|
|
/* Private data structures */
|
2019-03-23 15:08:58 -04:00
|
|
|
struct upd_printjob {
|
2020-11-29 09:42:05 -05:00
|
|
|
size_t jobsize;
|
|
|
|
int copies;
|
|
|
|
|
2018-06-15 14:55:03 -04:00
|
|
|
uint8_t *databuf;
|
|
|
|
int datalen;
|
2019-03-17 13:15:27 -04:00
|
|
|
|
|
|
|
uint16_t rows;
|
|
|
|
uint16_t cols;
|
|
|
|
uint32_t imglen;
|
2018-06-15 14:55:03 -04:00
|
|
|
};
|
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|
|
struct upd_ctx {
|
2020-08-11 20:27:26 -04:00
|
|
|
struct dyesub_connection *conn;
|
2013-07-14 15:55:35 -04:00
|
|
|
|
2019-03-17 13:15:27 -04:00
|
|
|
int native_bpp;
|
|
|
|
|
2020-12-01 22:47:40 -05:00
|
|
|
struct sony_updsts stsbuf;
|
|
|
|
struct sony_prints printbuf;
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2018-04-29 14:07:08 -04:00
|
|
|
struct marker marker;
|
2013-07-18 08:46:44 -04:00
|
|
|
};
|
2013-07-14 15:55:35 -04:00
|
|
|
|
2020-11-29 09:42:05 -05:00
|
|
|
static const char *upd_ribbons(int type, uint8_t code)
|
2013-07-18 08:46:44 -04:00
|
|
|
{
|
2020-11-28 09:35:24 -05:00
|
|
|
if (type == P_SONY_UPD895 || type == P_SONY_UPD897) {
|
|
|
|
return "UP-110 Roll";
|
2015-06-23 20:32:41 -04:00
|
|
|
}
|
2014-04-25 11:39:15 -04:00
|
|
|
|
2020-11-28 09:35:24 -05:00
|
|
|
/* CR10L/DR200/DR150 */
|
|
|
|
if (code == UPD_RIBBON_R206) {
|
|
|
|
return "R206 (8x6)";
|
2019-03-17 13:15:27 -04:00
|
|
|
}
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2020-11-28 09:35:24 -05:00
|
|
|
return "Unknown";
|
2018-06-15 14:55:03 -04:00
|
|
|
}
|
|
|
|
|
2020-12-01 22:47:40 -05:00
|
|
|
static int sonyupd_media_maxes(uint8_t type, uint8_t media)
|
|
|
|
{
|
|
|
|
if (type == P_SONY_UPDR150) {
|
|
|
|
if (media == UPD_RIBBON_R206)
|
|
|
|
return 350;
|
|
|
|
else
|
|
|
|
return 700; // XXX guess until we have more codes?
|
|
|
|
|
|
|
|
// XXX also differs for DR200 vs DR150?
|
|
|
|
} else if (type == P_SONY_UPCR10) {
|
|
|
|
return 300;
|
|
|
|
}
|
|
|
|
return CUPS_MARKER_UNAVAILABLE;
|
|
|
|
}
|
|
|
|
|
2019-04-16 09:53:54 -04:00
|
|
|
// UP-DR200
|
|
|
|
// 2UPC-R203 3.5x5 (770)
|
|
|
|
// 2UPC-R204 4x6 (700)
|
|
|
|
// 2UPC-R205 5x7 (400)
|
|
|
|
// 2UPC-R206 6x8 (350)
|
|
|
|
|
|
|
|
// UP-DR150
|
|
|
|
// 2UPC-R153 (610)
|
|
|
|
// 2UPC-R154 (550)
|
|
|
|
// 2UPC-R155 (335)
|
|
|
|
// 2UPC-R156 (295)
|
|
|
|
|
2020-11-28 10:39:28 -05:00
|
|
|
// UP-CR10L
|
|
|
|
|
|
|
|
// 2UPC-C13 (300)
|
|
|
|
// 2UPC-C14 (200)
|
|
|
|
// 2UPC-C15 (172)
|
|
|
|
|
2019-04-16 09:53:54 -04:00
|
|
|
// print order: ->YMCO->
|
|
|
|
// current prints (power on)
|
|
|
|
// total prints (lifetime)
|
|
|
|
// f/w version
|
|
|
|
|
2020-03-24 18:22:39 -04:00
|
|
|
static const char* upd895_statuses(uint8_t code)
|
2019-03-10 17:03:07 -04:00
|
|
|
{
|
|
|
|
switch (code) {
|
2019-04-16 14:26:51 -04:00
|
|
|
case UPD_STS1_IDLE:
|
2019-03-10 17:03:07 -04:00
|
|
|
return "Idle";
|
2019-04-16 14:26:51 -04:00
|
|
|
case UPD_STS1_DOOROPEN:
|
2019-03-10 17:03:07 -04:00
|
|
|
return "Door open";
|
2019-04-16 14:26:51 -04:00
|
|
|
case UPD_STS1_NOPAPER:
|
2019-03-10 17:03:07 -04:00
|
|
|
return "No paper";
|
2019-04-16 14:26:51 -04:00
|
|
|
case UPD_STS1_PRINTING:
|
2020-11-29 09:42:05 -05:00
|
|
|
case UPD_STS1_PRINTING2:
|
2019-04-16 14:26:51 -04:00
|
|
|
return "Printing";
|
2019-03-10 17:03:07 -04:00
|
|
|
default:
|
|
|
|
return "Unknown";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-28 10:39:28 -05:00
|
|
|
static const char* updr200_statuses(uint8_t code)
|
|
|
|
{
|
|
|
|
switch (code) {
|
|
|
|
case UPD_STS0_OK:
|
|
|
|
return "OK";
|
|
|
|
case UPD_STS0_DOOROPEN:
|
|
|
|
return "Door open";
|
|
|
|
case UPD_STS0_NOPAPER:
|
|
|
|
return "No paper";
|
|
|
|
case UPD_STS0_NORIBBON:
|
|
|
|
return "No ribbon";
|
|
|
|
default:
|
|
|
|
return "Unknown";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-28 09:35:24 -05:00
|
|
|
/* Now for the code */
|
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|
|
static int sony_get_status(struct upd_ctx *ctx, struct sony_updsts *buf)
|
2019-03-10 17:03:07 -04:00
|
|
|
{
|
|
|
|
int ret, num = 0;
|
2019-03-17 22:23:40 -04:00
|
|
|
uint8_t query[7] = { 0x1b, 0xe0, 0, 0, 0, 0x0f, 0 };
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPD895)
|
2019-03-17 22:23:40 -04:00
|
|
|
query[5] = 0x0e;
|
2019-03-15 22:57:13 -04:00
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
if ((ret = send_data(ctx->conn,
|
2019-03-10 17:03:07 -04:00
|
|
|
query, sizeof(query))))
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
ret = read_data(ctx->conn, (uint8_t*) buf, sizeof(*buf),
|
2019-03-17 13:15:27 -04:00
|
|
|
&num);
|
2019-03-10 17:03:07 -04:00
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
2019-03-15 22:57:13 -04:00
|
|
|
#if 0
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPD895 && ret != 14)
|
2019-03-15 22:57:13 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
else if (ret != 15)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
#endif
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2020-12-01 22:47:40 -05:00
|
|
|
buf->max_cols = be16_to_cpu(buf->max_cols);
|
|
|
|
buf->max_rows = be16_to_cpu(buf->max_rows);
|
|
|
|
|
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sony_get_prints(struct upd_ctx *ctx, struct sony_prints *buf)
|
|
|
|
{
|
|
|
|
int ret, num = 0;
|
|
|
|
uint8_t query[7] = { 0x1b, 0xef, 0, 0, 0, 0x06, 0 };
|
|
|
|
|
|
|
|
if ((ret = send_data(ctx->conn,
|
|
|
|
query, sizeof(query))))
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
|
|
|
ret = read_data(ctx->conn, (uint8_t*) buf, sizeof(*buf),
|
|
|
|
&num);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
|
|
|
buf->remain = be16_to_cpu(buf->remain);
|
2019-03-21 19:40:58 -04:00
|
|
|
|
2019-03-10 17:03:07 -04:00
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
2020-11-28 09:35:24 -05:00
|
|
|
static void* upd_init(void)
|
|
|
|
{
|
|
|
|
struct upd_ctx *ctx = malloc(sizeof(struct upd_ctx));
|
|
|
|
if (!ctx) {
|
|
|
|
ERROR("Memory Allocation Failure!\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
memset(ctx, 0, sizeof(struct upd_ctx));
|
|
|
|
return ctx;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upd_attach(void *vctx, struct dyesub_connection *conn, uint8_t jobid)
|
|
|
|
{
|
|
|
|
struct upd_ctx *ctx = vctx;
|
|
|
|
|
|
|
|
UNUSED(jobid);
|
|
|
|
|
|
|
|
ctx->conn = conn;
|
|
|
|
|
|
|
|
if (ctx->conn->type == P_SONY_UPD895 || ctx->conn->type == P_SONY_UPD897) {
|
|
|
|
ctx->marker.color = "#000000"; /* Ie black! */
|
|
|
|
ctx->native_bpp = 1;
|
|
|
|
} else {
|
|
|
|
ctx->marker.color = "#00FFFF#FF00FF#FFFF00";
|
|
|
|
ctx->native_bpp = 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (test_mode < TEST_MODE_NOATTACH) {
|
|
|
|
int ret;
|
|
|
|
if ((ret = sony_get_status(ctx, &ctx->stsbuf))) {
|
|
|
|
return ret;
|
|
|
|
}
|
2020-12-01 22:47:40 -05:00
|
|
|
if ((ctx->conn->type != P_SONY_UPD895 && ctx->conn->type != P_SONY_UPD897) && (ret = sony_get_prints(ctx, &ctx->printbuf))) {
|
|
|
|
return ret;
|
|
|
|
}
|
2020-11-28 09:35:24 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (test_mode >= TEST_MODE_NOATTACH && getenv("MEDIA_CODE")) {
|
|
|
|
ctx->marker.numtype = atoi(getenv("MEDIA_CODE"));
|
|
|
|
} else {
|
2020-12-01 22:47:40 -05:00
|
|
|
ctx->marker.numtype = ctx->stsbuf.ribbon;
|
2020-11-28 09:35:24 -05:00
|
|
|
}
|
|
|
|
|
2020-11-29 09:42:05 -05:00
|
|
|
ctx->marker.name = upd_ribbons(ctx->conn->type, ctx->stsbuf.ribbon);
|
2020-12-01 22:47:40 -05:00
|
|
|
if (test_mode >= TEST_MODE_NOATTACH || ctx->conn->type == P_SONY_UPD895 || ctx->conn->type == P_SONY_UPD897) {
|
|
|
|
ctx->marker.levelmax = CUPS_MARKER_UNAVAILABLE;
|
|
|
|
ctx->marker.levelnow = CUPS_MARKER_UNKNOWN;
|
|
|
|
} else {
|
2020-12-01 22:54:08 -05:00
|
|
|
ctx->marker.levelmax = sonyupd_media_maxes(ctx->conn->type, ctx->stsbuf.ribbon);
|
2020-12-01 22:47:40 -05:00
|
|
|
ctx->marker.levelnow = ctx->printbuf.remain;
|
|
|
|
}
|
|
|
|
|
2020-11-28 09:35:24 -05:00
|
|
|
|
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void upd_cleanup_job(const void *vjob)
|
|
|
|
{
|
|
|
|
const struct upd_printjob *job = vjob;
|
|
|
|
|
|
|
|
if (job->databuf)
|
|
|
|
free(job->databuf);
|
|
|
|
|
|
|
|
free((void*)job);
|
|
|
|
}
|
|
|
|
|
2018-09-29 20:50:37 -04:00
|
|
|
#define MAX_PRINTJOB_LEN (2048*2764*3 + 2048)
|
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|
|
static int upd_read_parse(void *vctx, const void **vjob, int data_fd, int copies) {
|
|
|
|
struct upd_ctx *ctx = vctx;
|
2014-02-11 13:41:15 -05:00
|
|
|
int len, run = 1;
|
2018-06-15 14:55:03 -04:00
|
|
|
uint32_t copies_offset = 0;
|
2019-03-19 12:52:40 -04:00
|
|
|
uint32_t param_offset = 0;
|
|
|
|
uint32_t data_offset = 0;
|
2018-06-15 14:55:03 -04:00
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|
|
struct upd_printjob *job = NULL;
|
2013-07-14 15:55:35 -04:00
|
|
|
|
2013-07-18 08:46:44 -04:00
|
|
|
if (!ctx)
|
2014-04-25 11:41:35 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-07-14 15:55:35 -04:00
|
|
|
|
2018-06-15 14:55:03 -04:00
|
|
|
job = malloc(sizeof(*job));
|
|
|
|
if (!job) {
|
|
|
|
ERROR("Memory allocation failure!\n");
|
|
|
|
return CUPS_BACKEND_RETRY_CURRENT;
|
2014-01-21 20:34:00 -05:00
|
|
|
}
|
2018-06-15 14:55:03 -04:00
|
|
|
memset(job, 0, sizeof(*job));
|
|
|
|
job->copies = copies;
|
2014-01-19 19:53:45 -05:00
|
|
|
|
2018-06-15 14:55:03 -04:00
|
|
|
job->datalen = 0;
|
|
|
|
job->databuf = malloc(MAX_PRINTJOB_LEN);
|
|
|
|
if (!job->databuf) {
|
2013-07-14 15:55:35 -04:00
|
|
|
ERROR("Memory allocation failure!\n");
|
2019-03-23 15:08:58 -04:00
|
|
|
upd_cleanup_job(job);
|
2018-02-16 10:49:21 -05:00
|
|
|
return CUPS_BACKEND_RETRY_CURRENT;
|
2013-07-14 15:55:35 -04:00
|
|
|
}
|
|
|
|
|
2014-01-19 19:53:45 -05:00
|
|
|
while(run) {
|
2014-02-11 13:41:15 -05:00
|
|
|
int i;
|
2014-01-19 19:53:45 -05:00
|
|
|
int keep = 0;
|
2018-06-15 14:55:03 -04:00
|
|
|
i = read(data_fd, job->databuf + job->datalen, 4);
|
2018-06-17 19:17:28 -04:00
|
|
|
if (i < 0) {
|
2019-03-23 15:08:58 -04:00
|
|
|
upd_cleanup_job(job);
|
2014-04-25 11:41:35 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
2018-06-17 19:17:28 -04:00
|
|
|
}
|
2014-01-20 19:41:52 -05:00
|
|
|
if (i == 0)
|
|
|
|
break;
|
2014-01-19 19:53:45 -05:00
|
|
|
|
2018-06-15 14:55:03 -04:00
|
|
|
memcpy(&len, job->databuf + job->datalen, sizeof(len));
|
2014-01-24 09:58:10 -05:00
|
|
|
len = le32_to_cpu(len);
|
2014-01-19 19:53:45 -05:00
|
|
|
|
|
|
|
/* Filter out chunks we don't send to the printer */
|
2018-08-13 17:57:52 -04:00
|
|
|
if (len & 0xf0000000) {
|
|
|
|
switch (len) {
|
|
|
|
case 0xfffffff3:
|
|
|
|
if(dyesub_debug)
|
|
|
|
DEBUG("Block ID '%08x' (len %d)\n", len, 0);
|
|
|
|
len = 0;
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPDR150)
|
2018-08-13 17:57:52 -04:00
|
|
|
run = 0;
|
|
|
|
break;
|
|
|
|
case 0xfffffff7:
|
|
|
|
if(dyesub_debug)
|
|
|
|
DEBUG("Block ID '%08x' (len %d)\n", len, 0);
|
|
|
|
len = 0;
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPCR10)
|
2018-08-13 17:57:52 -04:00
|
|
|
run = 0;
|
|
|
|
break;
|
|
|
|
case 0xfffffff8: // 895
|
|
|
|
case 0xfffffff4: // 897
|
|
|
|
if(dyesub_debug)
|
|
|
|
DEBUG("Block ID '%08x' (len %d)\n", len, 0);
|
|
|
|
len = 0;
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPD895 || ctx->conn->type == P_SONY_UPD897)
|
2018-08-13 17:57:52 -04:00
|
|
|
run = 0;
|
|
|
|
break;
|
2019-03-08 18:38:19 -05:00
|
|
|
case 0xffffff97:
|
|
|
|
if(dyesub_debug)
|
|
|
|
DEBUG("Block ID '%08x' (len %d)\n", len, 12);
|
|
|
|
len = 12;
|
|
|
|
break;
|
|
|
|
case 0xffffffef:
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPD895 || ctx->conn->type == P_SONY_UPD897) {
|
2019-03-08 18:38:19 -05:00
|
|
|
if(dyesub_debug)
|
|
|
|
DEBUG("Block ID '%08x' (len %d)\n", len, 0);
|
|
|
|
len = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* Intentional Fallthrough */
|
2018-08-13 17:57:52 -04:00
|
|
|
case 0xffffffeb:
|
|
|
|
case 0xffffffee:
|
|
|
|
case 0xfffffff5:
|
2014-01-19 19:53:45 -05:00
|
|
|
if(dyesub_debug)
|
2018-08-13 17:57:52 -04:00
|
|
|
DEBUG("Block ID '%08x' (len %d)\n", len, 4);
|
|
|
|
len = 4;
|
|
|
|
break;
|
2019-04-20 09:32:06 -04:00
|
|
|
case 0xffffffec:
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPD897) {
|
2019-04-20 09:44:15 -04:00
|
|
|
if(dyesub_debug)
|
|
|
|
DEBUG("Block ID '%08x' (len %d)\n", len, 4);
|
|
|
|
len = 4;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* Intentional Fallthrough */
|
2018-08-13 17:57:52 -04:00
|
|
|
default:
|
|
|
|
if(dyesub_debug)
|
|
|
|
DEBUG("Block ID '%08x' (len %d)\n", len, 0);
|
|
|
|
len = 0;
|
|
|
|
break;
|
2014-01-19 19:53:45 -05:00
|
|
|
}
|
2018-08-13 17:57:52 -04:00
|
|
|
} else {
|
|
|
|
/* Only keep these chunks */
|
|
|
|
if(dyesub_debug)
|
|
|
|
DEBUG("Data block (len %d)\n", len);
|
2019-03-08 18:38:19 -05:00
|
|
|
if (len > 0)
|
|
|
|
keep = 1;
|
2014-01-19 19:53:45 -05:00
|
|
|
}
|
|
|
|
if (keep)
|
2018-06-15 14:55:03 -04:00
|
|
|
job->datalen += sizeof(uint32_t);
|
2014-01-19 19:53:45 -05:00
|
|
|
|
2019-03-22 08:37:43 -04:00
|
|
|
/* Make sure we're not too large */
|
|
|
|
if (job->datalen + len > MAX_PRINTJOB_LEN) {
|
|
|
|
ERROR("Buffer overflow when parsing printjob! (%d+%d)\n",
|
|
|
|
job->datalen, len);
|
2019-05-06 21:10:43 -04:00
|
|
|
upd_cleanup_job(job);
|
2019-03-22 08:37:43 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
|
2014-01-19 19:53:45 -05:00
|
|
|
/* Read in the data chunk */
|
|
|
|
while(len > 0) {
|
2018-06-15 14:55:03 -04:00
|
|
|
i = read(data_fd, job->databuf + job->datalen, len);
|
2018-06-17 19:17:28 -04:00
|
|
|
if (i < 0) {
|
2019-03-23 15:08:58 -04:00
|
|
|
upd_cleanup_job(job);
|
2014-04-25 11:41:35 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
2018-06-17 19:17:28 -04:00
|
|
|
}
|
2014-01-26 15:05:02 -05:00
|
|
|
if (i == 0)
|
|
|
|
break;
|
2014-04-25 11:39:15 -04:00
|
|
|
|
2019-03-15 20:57:10 -04:00
|
|
|
/* Work out offset of copies command */
|
2019-03-17 13:15:27 -04:00
|
|
|
if (job->databuf[job->datalen] == 0x1b) {
|
2019-03-19 12:52:40 -04:00
|
|
|
int offset = 0;
|
|
|
|
if (i == 7)
|
|
|
|
offset = 4;
|
|
|
|
|
2019-03-17 13:15:27 -04:00
|
|
|
switch (job->databuf[job->datalen + 1]) {
|
2019-03-19 16:25:28 -04:00
|
|
|
case 0x15: /* Print dimensions */
|
|
|
|
param_offset = job->datalen + 16 + offset;
|
|
|
|
break;
|
2020-12-01 22:47:40 -05:00
|
|
|
// XXX case 0xc0:
|
|
|
|
// for param 03, take the value at offset 4 -- for (eg) 4x6 on 8x6 media, needs to be set to 0x02
|
2019-03-17 13:15:27 -04:00
|
|
|
case 0xee:
|
2019-03-19 12:52:40 -04:00
|
|
|
copies_offset = job->datalen + 7 + offset;
|
2019-03-17 13:15:27 -04:00
|
|
|
break;
|
2019-03-19 16:25:28 -04:00
|
|
|
case 0xe1: /* Image dimensions */
|
|
|
|
param_offset = job->datalen + 14 + offset;
|
2019-03-17 13:15:27 -04:00
|
|
|
break;
|
|
|
|
case 0xea:
|
2019-03-19 12:52:40 -04:00
|
|
|
data_offset = job->datalen + 6 + offset;
|
2019-03-17 13:15:27 -04:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2014-04-25 11:39:15 -04:00
|
|
|
}
|
|
|
|
|
2014-01-19 19:53:45 -05:00
|
|
|
if (keep)
|
2018-06-15 14:55:03 -04:00
|
|
|
job->datalen += i;
|
2014-01-19 19:53:45 -05:00
|
|
|
len -= i;
|
|
|
|
}
|
2013-07-14 15:55:35 -04:00
|
|
|
}
|
2018-06-17 19:17:28 -04:00
|
|
|
if (!job->datalen) {
|
2019-03-23 15:08:58 -04:00
|
|
|
upd_cleanup_job(job);
|
2014-04-25 11:41:35 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
2018-06-17 19:17:28 -04:00
|
|
|
}
|
2014-01-20 19:41:52 -05:00
|
|
|
|
2018-06-15 14:55:03 -04:00
|
|
|
/* Some models specify copies in the print job */
|
|
|
|
if (copies_offset) {
|
2019-10-19 20:24:21 -04:00
|
|
|
uint16_t tmp;
|
|
|
|
memcpy(&tmp, job->databuf + copies_offset, sizeof(tmp));
|
|
|
|
tmp = be16_to_cpu(tmp);
|
|
|
|
if (tmp < copies) { /* Use whichever one is larger */
|
|
|
|
tmp = cpu_to_be16(copies);
|
|
|
|
memcpy(job->databuf + copies_offset, &tmp, sizeof(tmp));
|
|
|
|
}
|
2018-06-15 14:55:03 -04:00
|
|
|
job->copies = 1;
|
|
|
|
}
|
|
|
|
|
2019-03-19 12:52:40 -04:00
|
|
|
/* Parse some other stuff */
|
|
|
|
if (param_offset) {
|
|
|
|
memcpy(&job->cols, job->databuf + param_offset, sizeof(uint16_t));
|
|
|
|
memcpy(&job->rows, job->databuf + param_offset + 2, sizeof(uint16_t));
|
|
|
|
job->cols = be16_to_cpu(job->cols);
|
|
|
|
job->rows = be16_to_cpu(job->rows);
|
|
|
|
}
|
|
|
|
if (data_offset) {
|
|
|
|
memcpy(&job->imglen, job->databuf + data_offset, sizeof(uint32_t));
|
|
|
|
job->imglen = be32_to_cpu(job->imglen);
|
|
|
|
}
|
|
|
|
|
2019-03-17 22:23:40 -04:00
|
|
|
/* Sanity check job parameters */
|
|
|
|
if (job->imglen != (uint32_t)(job->rows * job->cols * ctx->native_bpp))
|
|
|
|
{
|
2019-05-14 23:06:39 -04:00
|
|
|
ERROR("Job data length mismatch (%u vs %d)!\n",
|
2019-03-19 12:52:40 -04:00
|
|
|
job->imglen, job->rows * job->cols * ctx->native_bpp);
|
2019-05-26 13:05:41 -04:00
|
|
|
upd_cleanup_job(job);
|
2019-03-17 22:23:40 -04:00
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
|
2018-06-15 14:55:03 -04:00
|
|
|
*vjob = job;
|
|
|
|
|
2014-04-25 11:41:35 -04:00
|
|
|
return CUPS_BACKEND_OK;
|
2013-07-18 08:46:44 -04:00
|
|
|
}
|
2013-07-14 15:55:35 -04:00
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|
|
static int upd_main_loop(void *vctx, const void *vjob) {
|
|
|
|
struct upd_ctx *ctx = vctx;
|
2017-08-05 18:46:55 -04:00
|
|
|
int i, ret;
|
2018-06-15 14:55:03 -04:00
|
|
|
int copies;
|
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|
|
const struct upd_printjob *job = vjob;
|
2013-07-14 15:55:35 -04:00
|
|
|
|
2013-07-18 08:46:44 -04:00
|
|
|
if (!ctx)
|
2014-04-25 11:41:35 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2018-06-15 14:55:03 -04:00
|
|
|
if (!job)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
2013-07-14 15:55:35 -04:00
|
|
|
|
2018-06-15 14:55:03 -04:00
|
|
|
copies = job->copies;
|
2014-04-25 11:39:15 -04:00
|
|
|
|
2013-07-14 15:55:35 -04:00
|
|
|
top:
|
2019-03-17 13:15:27 -04:00
|
|
|
/* Send Unknown CMD. Resets? */
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPD897) {
|
2019-03-17 13:15:27 -04:00
|
|
|
const uint8_t cmdbuf[7] = { 0x1b, 0x1f, 0, 0, 0, 0, 0 };
|
2020-08-11 20:27:26 -04:00
|
|
|
ret = send_data(ctx->conn,
|
2019-03-17 13:15:27 -04:00
|
|
|
cmdbuf, sizeof(cmdbuf));
|
2019-03-10 17:03:07 -04:00
|
|
|
if (ret)
|
2019-03-17 13:15:27 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
}
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2019-03-17 13:15:27 -04:00
|
|
|
/* Query printer status */
|
|
|
|
ret = sony_get_status(ctx, &ctx->stsbuf);
|
|
|
|
|
|
|
|
if (ret)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
|
|
|
/* Sanity check job parameters */
|
|
|
|
if (job->rows > ctx->stsbuf.max_rows ||
|
|
|
|
job->cols > ctx->stsbuf.max_cols) {
|
|
|
|
ERROR("Job dimensions (%u/%u) exceed printer max (%u/%u)\n",
|
|
|
|
job->cols, job->rows,
|
|
|
|
ctx->stsbuf.max_cols,
|
|
|
|
ctx->stsbuf.max_rows);
|
|
|
|
return CUPS_BACKEND_CANCEL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check for idle */
|
2020-11-23 19:52:02 -05:00
|
|
|
if (ctx->stsbuf.sts1 != UPD_STS1_IDLE) {
|
|
|
|
if (ctx->stsbuf.sts1 == UPD_STS1_PRINTING) {
|
2019-03-17 13:15:27 -04:00
|
|
|
INFO("Waiting for printer idle...\n");
|
|
|
|
sleep(1);
|
|
|
|
goto top;
|
2020-11-23 19:52:02 -05:00
|
|
|
} else {
|
|
|
|
// XXX some sort of error?
|
2019-03-10 17:03:07 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-17 13:15:27 -04:00
|
|
|
/* Send RESET */
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type != P_SONY_UPD895) {
|
2019-03-17 13:15:27 -04:00
|
|
|
const uint8_t rstbuf[7] = { 0x1b, 0x16, 0, 0, 0, 0, 0 };
|
2020-08-11 20:27:26 -04:00
|
|
|
ret = send_data(ctx->conn,
|
2019-03-17 13:15:27 -04:00
|
|
|
rstbuf, sizeof(rstbuf));
|
|
|
|
if (ret)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0 /* Unknown query */
|
2020-08-11 20:27:26 -04:00
|
|
|
if (ctx->conn->type == P_SONY_UPD897) {
|
2019-03-17 13:15:27 -04:00
|
|
|
// -> 1b e6 00 00 00 08 00
|
|
|
|
// <- ???
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Send over job */
|
2017-08-05 18:46:55 -04:00
|
|
|
i = 0;
|
2018-06-15 14:55:03 -04:00
|
|
|
while (i < job->datalen) {
|
2014-01-26 15:05:02 -05:00
|
|
|
uint32_t len;
|
2018-06-15 14:55:03 -04:00
|
|
|
memcpy(&len, job->databuf + i, sizeof(len));
|
2014-01-26 15:05:02 -05:00
|
|
|
len = le32_to_cpu(len);
|
2014-01-19 19:53:45 -05:00
|
|
|
|
|
|
|
i += sizeof(uint32_t);
|
|
|
|
|
2020-08-11 20:27:26 -04:00
|
|
|
if ((ret = send_data(ctx->conn,
|
2018-06-15 14:55:03 -04:00
|
|
|
job->databuf + i, len)))
|
2014-04-25 11:41:35 -04:00
|
|
|
return CUPS_BACKEND_FAILED;
|
2014-01-26 15:05:02 -05:00
|
|
|
|
2014-01-19 19:53:45 -05:00
|
|
|
i += len;
|
2013-07-14 15:55:35 -04:00
|
|
|
}
|
|
|
|
|
2019-03-17 13:15:27 -04:00
|
|
|
// XXX generate and send copy cmd instead of using the offset.
|
|
|
|
// 1b ee 00 00 00 02 00 NN NN (BE)
|
|
|
|
|
2019-03-17 22:23:40 -04:00
|
|
|
/* Wait for completion! */
|
|
|
|
retry:
|
|
|
|
sleep(1);
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2019-03-17 22:23:40 -04:00
|
|
|
/* Check for idle */
|
|
|
|
ret = sony_get_status(ctx, &ctx->stsbuf);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2019-03-17 22:23:40 -04:00
|
|
|
switch (ctx->stsbuf.sts1) {
|
2019-04-16 14:26:51 -04:00
|
|
|
case UPD_STS1_IDLE:
|
2019-03-17 22:23:40 -04:00
|
|
|
goto done;
|
2019-04-16 14:26:51 -04:00
|
|
|
case UPD_STS1_PRINTING:
|
2020-11-29 09:42:05 -05:00
|
|
|
case UPD_STS1_PRINTING2:
|
2019-03-17 22:23:40 -04:00
|
|
|
break;
|
|
|
|
default:
|
2020-11-28 10:39:28 -05:00
|
|
|
if (ctx->conn->type == P_SONY_UPD895 || ctx->conn->type == P_SONY_UPD897) {
|
|
|
|
ERROR("Printer error: %s (%02x)\n", upd895_statuses(ctx->stsbuf.sts1), ctx->stsbuf.sts1);
|
|
|
|
} else {
|
|
|
|
ERROR("Printer error: %s (%02x)\n", updr200_statuses(ctx->stsbuf.sts0), ctx->stsbuf.sts0);
|
|
|
|
}
|
2019-03-17 22:23:40 -04:00
|
|
|
return CUPS_BACKEND_STOP;
|
|
|
|
}
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2019-04-16 14:26:51 -04:00
|
|
|
if (fast_return && ctx->stsbuf.printing != UPD_PRINTING_IDLE) {
|
2019-03-17 22:23:40 -04:00
|
|
|
INFO("Fast return mode enabled.\n");
|
|
|
|
} else {
|
|
|
|
goto retry;
|
2019-03-10 17:03:07 -04:00
|
|
|
}
|
|
|
|
|
2013-07-14 15:55:35 -04:00
|
|
|
/* Clean up */
|
|
|
|
if (terminate)
|
|
|
|
copies = 1;
|
|
|
|
|
2019-03-10 17:03:07 -04:00
|
|
|
done:
|
2014-01-22 09:10:34 -05:00
|
|
|
INFO("Print complete (%d copies remaining)\n", copies - 1);
|
2013-07-14 15:55:35 -04:00
|
|
|
|
|
|
|
if (copies && --copies) {
|
|
|
|
goto top;
|
|
|
|
}
|
|
|
|
|
2014-04-25 11:41:35 -04:00
|
|
|
return CUPS_BACKEND_OK;
|
2013-07-14 15:55:35 -04:00
|
|
|
}
|
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|
|
static int upd895_dump_status(struct upd_ctx *ctx)
|
2019-03-10 17:03:07 -04:00
|
|
|
{
|
2019-03-17 13:15:27 -04:00
|
|
|
int ret = sony_get_status(ctx, &ctx->stsbuf);
|
2019-03-10 17:03:07 -04:00
|
|
|
if (ret < 0)
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
|
2020-12-01 22:54:08 -05:00
|
|
|
if (ctx->conn->type != P_SONY_UPD895 && ctx->conn->type != P_SONY_UPD897 && (ret = sony_get_prints(ctx, &ctx->printbuf))) {
|
|
|
|
return CUPS_BACKEND_FAILED;
|
|
|
|
}
|
|
|
|
|
2020-11-28 10:39:28 -05:00
|
|
|
if (ctx->conn->type == P_SONY_UPD895 || ctx->conn->type == P_SONY_UPD897) {
|
|
|
|
INFO("Printer status: %s (%02x)\n", upd895_statuses(ctx->stsbuf.sts1), ctx->stsbuf.sts1);
|
|
|
|
} else {
|
|
|
|
INFO("Printer status: %s (%02x)\n", updr200_statuses(ctx->stsbuf.sts0), ctx->stsbuf.sts0);
|
|
|
|
}
|
|
|
|
|
2019-04-16 14:26:51 -04:00
|
|
|
if (ctx->stsbuf.printing != UPD_PRINTING_IDLE &&
|
|
|
|
ctx->stsbuf.sts1 == UPD_STS1_PRINTING)
|
2020-12-01 22:54:08 -05:00
|
|
|
INFO("Remaining copies to print: %d\n", ctx->stsbuf.remain);
|
2019-03-10 17:03:07 -04:00
|
|
|
|
2020-11-29 09:42:05 -05:00
|
|
|
INFO("Media: %s (%02x)\n", upd_ribbons(ctx->conn->type, ctx->stsbuf.ribbon), ctx->stsbuf.ribbon);
|
2020-11-28 09:35:24 -05:00
|
|
|
|
2020-12-01 22:54:08 -05:00
|
|
|
if (ctx->conn->type != P_SONY_UPD895 && ctx->conn->type != P_SONY_UPD897) {
|
2020-12-10 17:48:06 -05:00
|
|
|
INFO("Media remaining: %d/%d\n", ctx->printbuf.remain, sonyupd_media_maxes(ctx->conn->type, ctx->stsbuf.ribbon));
|
2020-12-01 22:54:08 -05:00
|
|
|
}
|
2019-03-10 17:03:07 -04:00
|
|
|
return CUPS_BACKEND_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-03-23 15:08:58 -04:00
|
|