/* * Mitsubishi CP-9xxx Photo Printer Family CUPS backend * * (c) 2014-2020 Solomon Peachy * * The latest version of this program can be found at: * * https://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 * along with this program; if not, see . * * SPDX-License-Identifier: GPL-3.0+ * */ #define BACKEND mitsu9550_backend #include "backend_common.h" #include "backend_mitsu.h" #define MITSU_M98xx_LAMINATE_FILE "M98MATTE.raw" #define MITSU_M98xx_DATATABLE_FILE "M98TABLE.dat" #define MITSU_M98xx_LUT_FILE "M98XXL01.lut" #define MITSU_CP30D_LUT_FILE "CP30LT_1.lut" #define LAMINATE_STRIDE 1868 /* Spool file structures */ /* Print parameters1 */ struct mitsu9550_hdr1 { uint8_t cmd[4]; /* 1b 57 20 2e */ uint8_t unk[10]; /* 00 0a 10 00 [...] */ uint16_t cols; /* BE */ uint16_t rows; /* BE */ uint8_t matte; /* CP9810/9820 only. 01 for matte, 00 glossy */ uint8_t null[31]; } __attribute__((packed)); /* Print parameters2 */ struct mitsu9550_hdr2 { uint8_t cmd[4]; /* 1b 57 21 2e */ uint8_t unk[24]; /* 00 80 00 22 08 03 [...] */ uint16_t copies; /* BE, 1-680 */ uint8_t null[2]; uint8_t cut; /* 00 == normal, 83 == 2x6*2 */ uint8_t unkb[5]; uint8_t mode; /* 00 == fine, 80 == superfine */ uint8_t unkc[11]; /* 00 [...] 00 01 ; note [7][8][9] are cp98xx extensions for sharpness/reversed/lut */ } __attribute__((packed)); /* Fine Deep selection (9550 only) */ struct mitsu9550_hdr3 { uint8_t cmd[4]; /* 1b 57 22 2e */ uint8_t unk[7]; /* 00 40 00 [...] */ uint8_t mode2; /* 00 == normal, 01 == finedeep */ uint8_t null[38]; } __attribute__((packed)); /* Error policy? */ struct mitsu9550_hdr4 { uint8_t cmd[4]; /* 1b 57 26 2e */ uint8_t unk[46]; /* 00 70 00 00 00 00 00 00 01 01 00 [...] */ } __attribute__((packed)); /* Data plane header */ struct mitsu9550_plane { uint8_t cmd[4]; /* 1b 5a 54 XX */ /* XX == 0x10 if 16bpp, 0x00 for 8bpp */ uint16_t col_offset; /* BE, normally 0, where we start dumping data */ uint16_t row_offset; /* BE, normally 0, where we start dumping data */ uint16_t cols; /* BE */ uint16_t rows; /* BE */ } __attribute__((packed)); /* Command header */ struct mitsu9550_cmd { uint8_t cmd[4]; } __attribute__((packed)); /* Private data structure */ struct mitsu9550_printjob { size_t jobsize; int copies; uint8_t *databuf; uint32_t datalen; uint16_t rows; uint16_t cols; uint32_t plane_len; int is_raw; /* Parse headers separately */ struct mitsu9550_hdr1 hdr1; int hdr1_present; struct mitsu9550_hdr2 hdr2; int hdr2_present; struct mitsu9550_hdr3 hdr3; int hdr3_present; struct mitsu9550_hdr4 hdr4; int hdr4_present; }; struct mitsu9550_ctx { struct dyesub_connection *conn; int is_s; int is_98xx; int need_lib; int footer_len; const char *lut_fname; char fwver[7]; /* 6 + null */ struct marker marker; /* CP98xx stuff */ struct mitsu_lib lib; const struct mitsu98xx_data *m98xxdata; }; /* Printer data structures */ #define CP9XXX_STS_x20 0x20 #define CP9XXX_STS_x21 0x21 /* struct mitsu9550_status2 */ #define CP9XXX_STS_x22 0x22 #define CP9XXX_STS_FWVER 0x23 #define CP9XXX_STS_MEDIA 0x24 /* struct mitsu9550_media */ #define CP9XXX_STS_x26 0x26 #define CP9XXX_STS_x30 0x30 /* struct mitsu9550_status */ #define CP9XXX_STS_x32 0x32 /* struct mitsucp30_status */ struct mitsu9550_media { uint8_t hdr[2]; /* 24 2e */ uint8_t unk[12]; uint8_t type; uint8_t unka[13]; uint16_t max; /* BE, prints per media */ uint16_t remain2; /* BE, prints remaining (CP30)*/ uint16_t remain; /* BE, prints remaining (Everything else) */ uint8_t unkc[14]; } __attribute__((packed)); struct mitsu9550_status { uint8_t hdr[2]; /* 30 2e */ uint8_t null[4]; uint8_t sts1; // MM uint8_t nullb[1]; uint16_t copies; // BE, NN uint8_t sts2; // ZZ (9600 only?) uint8_t nullc[5]; uint8_t sts3; // QQ uint8_t sts4; // RR uint8_t sts5; // SS uint8_t nulld[25]; uint8_t sts6; // TT uint8_t sts7; // UU uint8_t nulle[2]; } __attribute__((packed)); struct mitsucp30_status { uint8_t hdr[2]; /* 32 2e */ uint8_t zero[2]; uint16_t serno[13]; /* UTF16, BE */ uint8_t sts; /* CP30_STS_* */ uint8_t sts2; uint8_t zerob; uint16_t err; /* CP30_ERR_* */ uint8_t zeroc[2]; uint8_t remain; /* on media */ uint8_t zerod[9]; } __attribute__((packed)); #define CP30_STS_IDLE 0x00 #define CP30_STS_PRINT 0x20 #define CP30_STS_PRINT_A 0x10 //Load #define CP30_STS_PRINT_B 0x20 //Y #define CP30_STS_PRINT_C 0x30 //M #define CP30_STS_PRINT_D 0x40 //C #define CP30_STS_PRINT_E 0x60 //Eject? #define CP30_ERR_OK 0x0000 #define CP30_ERR_NOPC 0x0303 #define CP30_ERR_NORIBBON 0x0101 #define CP30_ERR_TRAYFULL 0x1404 struct mitsu9550_status2 { uint8_t hdr[2]; /* 21 2e */ uint8_t unk[40]; uint16_t remain; /* BE, media remaining */ uint8_t unkb[4]; /* 0a 00 00 01 */ } __attribute__((packed)); static int mitsu9550_main_loop(void *vctx, const void *vjob); static const char *cp30_errors(uint16_t err) { switch(err){ case CP30_ERR_OK: return "None"; case CP30_ERR_NOPC: return "No Paper Cassette"; case CP30_ERR_NORIBBON: return "No Ribbon Loaded"; case CP30_ERR_TRAYFULL: return "Output Tray Full"; default: return "Unknown"; } } static const char *cp30_media_types(uint16_t remain) { switch(remain) { case 80: return "CK30S"; case 50: return "CK30L"; default: return "Unknown"; } } #define CMDBUF_LEN 64 #define READBACK_LEN 128 #define QUERY_STATUS_I \ struct mitsu9550_status *sts = (struct mitsu9550_status*) rdbuf; \ struct mitsucp30_status *sts30 = (struct mitsucp30_status*) rdbuf; \ struct mitsu9550_media *media = (struct mitsu9550_media *) rdbuf; \ uint16_t donor; \ /* media */ \ ret = mitsu9550_get_status(ctx, rdbuf, CP9XXX_STS_MEDIA); \ if (ret < 0) \ return CUPS_BACKEND_FAILED; \ \ if (ctx->conn->type == P_MITSU_CP30D) { \ donor = be16_to_cpu(media->remain2); \ } else { \ donor = be16_to_cpu(media->remain); \ } \ if (donor != ctx->marker.levelnow) { \ ctx->marker.levelnow = donor; \ dump_markers(&ctx->marker, 1, 0); \ } \ /* Sanity-check media response */ \ if ((media->remain == 0 && media->remain2 == 0) || media->max == 0) { \ ERROR("Printer out of media!\n"); \ return CUPS_BACKEND_HOLD; \ } \ #define QUERY_STATUS_II \ if (ctx->conn->type != P_MITSU_CP30D) { \ /* struct mitsu9550_status2 *sts2 = (struct mitsu9550_status2*) rdbuf; */ \ ret = mitsu9550_get_status(ctx, rdbuf, CP9XXX_STS_x21); \ if (ret < 0) \ return CUPS_BACKEND_FAILED; \ /* XXX validate status2 ? */ \ } #define QUERY_STATUS_III \ /* Check for known errors */ \ if (sts->sts2 != 0) { \ ERROR("Printer cover open!\n"); \ return CUPS_BACKEND_STOP; \ } #define QUERY_STATUS_IIIB \ /* Check for known errors */ \ if (sts30->err != CP30_ERR_OK) { \ ERROR("%s (%04x)!\n", cp30_errors(be16_to_cpu(sts30->err)), be16_to_cpu(sts30->err)); \ return CUPS_BACKEND_STOP; \ } #define QUERY_STATUS_IV \ if (ctx->conn->type == P_MITSU_CP30D) { \ ret = mitsu9550_get_status(ctx, rdbuf, CP9XXX_STS_x32); \ if (ret < 0) \ return CUPS_BACKEND_FAILED; \ \ if (sts30->sts != CP30_STS_IDLE) { \ sleep(1); \ goto top; \ } \ QUERY_STATUS_IIIB; \ } else { \ ret = mitsu9550_get_status(ctx, rdbuf, CP9XXX_STS_x30); \ if (ret < 0) \ return CUPS_BACKEND_FAILED; \ \ /* Make sure we're idle */ \ if (sts->sts5 != 0) { \ sleep(1); \ goto top; \ } \ QUERY_STATUS_III; \ } \ static int mitsu98xx_fillmatte(struct mitsu9550_printjob *job) { int ret; /* Fill in the lamination plane header */ struct mitsu9550_plane *matte = (struct mitsu9550_plane *)(job->databuf + job->datalen); matte->cmd[0] = 0x1b; matte->cmd[1] = 0x5a; matte->cmd[2] = 0x54; matte->cmd[3] = 0x10; matte->row_offset = 0; matte->col_offset = 0; matte->cols = cpu_to_be16(job->hdr1.cols); matte->rows = cpu_to_be16(job->hdr1.rows); job->datalen += sizeof(struct mitsu9550_plane); ret = mitsu_readlamdata(MITSU_M98xx_LAMINATE_FILE, LAMINATE_STRIDE, job->databuf, &job->datalen, job->rows, job->cols, 2); if (ret) return ret; /* Fill in the lamination plane footer */ job->databuf[job->datalen++] = 0x1b; job->databuf[job->datalen++] = 0x50; job->databuf[job->datalen++] = 0x56; job->databuf[job->datalen++] = 0x00; return CUPS_BACKEND_OK; } static int mitsu9550_get_status(struct mitsu9550_ctx *ctx, uint8_t *resp, int type); static const char *mitsu9550_media_types(uint8_t type, uint8_t is_s); static void *mitsu9550_init(void) { struct mitsu9550_ctx *ctx = malloc(sizeof(struct mitsu9550_ctx)); if (!ctx) { ERROR("Memory Allocation Failure!\n"); return NULL; } memset(ctx, 0, sizeof(struct mitsu9550_ctx)); return ctx; } static int mitsu9550_attach(void *vctx, struct dyesub_connection *conn, uint8_t jobid) { struct mitsu9550_ctx *ctx = vctx; struct mitsu9550_media media; UNUSED(jobid); ctx->conn = conn; if (ctx->conn->type == P_MITSU_9550S || ctx->conn->type == P_MITSU_9800S) ctx->is_s = 1; if (ctx->conn->type == P_MITSU_9800 || ctx->conn->type == P_MITSU_9800S || ctx->conn->type == P_MITSU_9810) { ctx->is_98xx = 1; ctx->need_lib = 1; ctx->lut_fname = MITSU_M98xx_LUT_FILE; } if (ctx->conn->type == P_MITSU_CP30D) { ctx->need_lib = 1; } if (ctx->need_lib) { #if defined(WITH_DYNAMIC) /* Attempt to open the library */ if (mitsu_loadlib(&ctx->lib, ctx->conn->type)) #endif WARNING("Dynamic library support not loaded, will be unable to print."); } if (ctx->conn->type == P_MITSU_CP30D) { ctx->footer_len = 6; ctx->lut_fname = MITSU_CP30D_LUT_FILE; } else { ctx->footer_len = 4; } if (test_mode < TEST_MODE_NOATTACH) { uint8_t buf[48]; if (mitsu9550_get_status(ctx, (uint8_t*) &media, CP9XXX_STS_MEDIA)) return CUPS_BACKEND_FAILED; /* Get FW Version */ if (mitsu9550_get_status(ctx, buf, CP9XXX_STS_FWVER)) return CUPS_BACKEND_FAILED; memcpy(ctx->fwver, &buf[6], 6); ctx->fwver[6] = 0; // XXX get serial number too? } else { int media_code = 0x2; if (getenv("MEDIA_CODE")) media_code = atoi(getenv("MEDIA_CODE")) & 0xf; media.max = cpu_to_be16(400); media.remain = cpu_to_be16(330); media.remain2 = cpu_to_be16(330); media.type = media_code; ctx->fwver[0] = 0; } ctx->marker.color = "#00FFFF#FF00FF#FFFF00"; ctx->marker.numtype = media.type; ctx->marker.levelmax = be16_to_cpu(media.max); if (ctx->conn->type == P_MITSU_CP30D) { ctx->marker.name = cp30_media_types(be16_to_cpu(media.max)); ctx->marker.levelnow = be16_to_cpu(media.remain2); } else { ctx->marker.name = mitsu9550_media_types(media.type, ctx->is_s); ctx->marker.levelnow = be16_to_cpu(media.remain); } return CUPS_BACKEND_OK; } static void mitsu9550_cleanup_job(const void *vjob) { const struct mitsu9550_printjob *job = vjob; if (job->databuf) free(job->databuf); free((void*)job); } static void mitsu9550_teardown(void *vctx) { struct mitsu9550_ctx *ctx = vctx; if (!ctx) return; if (ctx->m98xxdata) ctx->lib.CP98xx_DestroyData(ctx->m98xxdata); mitsu_destroylib(&ctx->lib); free(ctx); } static int mitsu9550_read_parse(void *vctx, const void **vjob, int data_fd, int copies) { struct mitsu9550_ctx *ctx = vctx; uint8_t buf[sizeof(struct mitsu9550_hdr1)]; int remain, i; uint32_t planelen = 0; struct mitsu9550_printjob *job = NULL; if (!ctx) return CUPS_BACKEND_FAILED; job = malloc(sizeof(*job)); if (!job) { ERROR("Memory allocation failure!\n"); return CUPS_BACKEND_RETRY_CURRENT; } memset(job, 0, sizeof(*job)); job->is_raw = 1; job->jobsize = sizeof(*job); job->copies = copies; top: /* Read in initial header */ remain = sizeof(buf); while (remain > 0) { i = read(data_fd, buf + sizeof(buf) - remain, remain); if (i == 0) { mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } if (i < 0) { mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } remain -= i; } /* Sanity check */ if (buf[0] != 0x1b || buf[1] != 0x57 || buf[3] != 0x2e) { if (!job->hdr1_present || !job->hdr2_present) { ERROR("Unrecognized data format (%02x%02x%02x%02x)!\n", buf[0], buf[1], buf[2], buf[3]); mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } else if (buf[0] == 0x1b && buf[1] == 0x5a && buf[2] == 0x54) { /* We're in the data portion now */ if (buf[3] == 0x10) planelen *= 2; else if (ctx->is_98xx && buf[3] == 0x80) job->is_raw = 0; goto hdr_done; } else { ERROR("Unrecognized data block (%02x%02x%02x%02x)!\n", buf[0], buf[1], buf[2], buf[3]); mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } } switch(buf[2]) { case 0x20: /* header 1 */ memcpy(&job->hdr1, buf, sizeof(job->hdr1)); job->hdr1_present = 1; /* Work out printjob size */ job->rows = be16_to_cpu(job->hdr1.rows); job->cols = be16_to_cpu(job->hdr1.cols); planelen = job->rows * job->cols; break; case 0x21: /* header 2 */ memcpy(&job->hdr2, buf, sizeof(job->hdr2)); job->hdr2_present = 1; break; case 0x22: /* header 3 */ memcpy(&job->hdr3, buf, sizeof(job->hdr3)); job->hdr3_present = 1; break; case 0x26: /* header 4 */ memcpy(&job->hdr4, buf, sizeof(job->hdr4)); job->hdr4_present = 1; break; default: ERROR("Unrecognized header format (%02x)!\n", buf[2]); mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } /* Read in the next chunk */ goto top; hdr_done: /* Read in CP98xx data tables if necessary */ if (ctx->is_98xx && !job->is_raw && !ctx->m98xxdata) { char full[2048]; if (!ctx->lib.dl_handle) { ERROR("!!! Image Processing Library not found, aborting!\n"); mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } snprintf(full, sizeof(full), "%s/%s", corrtable_path, MITSU_M98xx_DATATABLE_FILE); DEBUG("Reading in 98xx data from disk\n"); ctx->m98xxdata = ctx->lib.CP98xx_GetData(full); if (!ctx->m98xxdata) { ERROR("Unable to read 98xx data table file '%s'\n", full); } } if (job->is_raw) { /* We have three planes + headers and the final terminator to read */ remain = 3 * (planelen + sizeof(struct mitsu9550_plane)) + ctx->footer_len; } else { /* We have one plane + header and the final terminator to read */ remain = planelen * 3 + sizeof(struct mitsu9550_plane) + ctx->footer_len; } /* Mitsu9600 windows spool uses more, smaller blocks, but plane data is the same */ if (ctx->conn->type == P_MITSU_9600) { remain += 128 * sizeof(struct mitsu9550_plane); /* 39 extra seen on 4x6" */ } /* 9550S/9800S doesn't typically sent over hdr4! */ if (ctx->conn->type == P_MITSU_9550S || ctx->conn->type == P_MITSU_9800S) { /* XXX Has to do with error policy, but not sure what. Mitsu9550-S/9800-S will set this based on a command, but it's not part of the standard job spool */ job->hdr4_present = 0; } /* Disable matte if the printer doesn't support it */ if (job->hdr1.matte) { if (ctx->conn->type != P_MITSU_9810) { WARNING("Matte not supported on this printer, disabling\n"); job->hdr1.matte = 0; } else if (job->is_raw) { remain += planelen + sizeof(struct mitsu9550_plane) + sizeof(struct mitsu9550_cmd); } } /* Allocate buffer for the payload */ job->datalen = 0; job->databuf = malloc(remain); if (!job->databuf) { ERROR("Memory allocation failure!\n"); mitsu9550_cleanup_job(job); return CUPS_BACKEND_RETRY_CURRENT; } /* Load up the data blocks.*/ while(1) { /* Note that 'buf' needs to be already filled here! */ struct mitsu9550_plane *plane = (struct mitsu9550_plane *)buf; /* Sanity check header... */ if (plane->cmd[0] != 0x1b || plane->cmd[1] != 0x5a || plane->cmd[2] != 0x54) { ERROR("Unrecognized data read @%d (%02x%02x%02x%02x)!\n", job->datalen, plane->cmd[0], plane->cmd[1], plane->cmd[2], plane->cmd[3]); mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } /* Work out the length of this block */ planelen = be16_to_cpu(plane->rows) * be16_to_cpu(plane->cols); if (plane->cmd[3] == 0x10) planelen *= 2; if (plane->cmd[3] == 0x80) planelen *= 3; /* Copy plane header into buffer */ memcpy(job->databuf + job->datalen, buf, sizeof(buf)); job->datalen += sizeof(buf); planelen -= sizeof(buf) - sizeof(struct mitsu9550_plane); /* Read in the spool data */ while(planelen > 0) { i = read(data_fd, job->databuf + job->datalen, planelen); if (i == 0) { mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } if (i < 0) { mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } job->datalen += i; planelen -= i; } /* Try to read in the next chunk. It will be one of: - Additional block header (12B) - Job footer (4B) */ i = read(data_fd, buf, ctx->footer_len); if (i == 0) { mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } if (i < 0) { mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } /* Is this a "job end" marker? */ if (plane->cmd[0] == 0x1b && plane->cmd[1] == 0x50 && plane->cmd[3] == 0x00) { /* store it in the buffer */ memcpy(job->databuf + job->datalen, buf, ctx->footer_len); job->datalen += ctx->footer_len; /* Unless we have a raw matte plane following, we're done */ if (job->hdr1.matte != 0x01 || !job->is_raw) break; remain = sizeof(buf); } else { /* It's part of a block header, mark what we've read */ remain = sizeof(buf) - ctx->footer_len; } /* Read in the rest of the header */ while (remain > 0) { i = read(data_fd, buf + sizeof(buf) - remain, remain); if (i == 0) { mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } if (i < 0) { mitsu9550_cleanup_job(job); return CUPS_BACKEND_CANCEL; } remain -= i; } } /* One bit of fixup */ if (ctx->conn->type == P_MITSU_CP30D) job->is_raw = 0; /* Apply LUT, if job calls for it.. */ if (ctx->lut_fname && !job->is_raw && job->hdr2.unkc[9]) { int ret; if (ctx->conn->type == P_MITSU_CP30D) { uint32_t planelen = job->rows * job->cols + sizeof(struct mitsu9550_plane); ret = mitsu_apply3dlut_plane(&ctx->lib, ctx->lut_fname, job->databuf + sizeof(struct mitsu9550_plane), job->databuf + (planelen + sizeof(struct mitsu9550_plane)), job->databuf + (planelen * 2 + sizeof(struct mitsu9550_plane)), job->cols, job->rows); } else { ret = mitsu_apply3dlut_packed(&ctx->lib, ctx->lut_fname, job->databuf + sizeof(struct mitsu9550_plane), job->cols, job->rows, job->cols * 3, COLORCONV_BGR); } if (ret) { mitsu9550_cleanup_job(job); return ret; } job->hdr2.unkc[9] = 0; } /* Update printjob header to reflect number of requested copies */ if (job->hdr2_present) { if (be16_to_cpu(job->hdr2.copies) < copies) job->hdr2.copies = cpu_to_be16(copies); copies = 1; } job->copies = copies; /* All further work is in main loop */ if (test_mode >= TEST_MODE_NOPRINT) mitsu9550_main_loop(ctx, job); *vjob = job; return CUPS_BACKEND_OK; } static int mitsu9550_get_status(struct mitsu9550_ctx *ctx, uint8_t *resp, int type) { struct mitsu9550_cmd cmd; int num, ret; /* Send Printer Query */ cmd.cmd[0] = 0x1b; cmd.cmd[1] = 0x56; // XXX need version for 0x72 cmd.cmd[2] = type; cmd.cmd[3] = 0x00; if ((ret = send_data(ctx->conn, (uint8_t*) &cmd, sizeof(cmd)))) return ret; ret = read_data(ctx->conn, resp, sizeof(struct mitsu9550_status), &num); if (ret < 0) return ret; if (num != sizeof(struct mitsu9550_status)) { ERROR("Short Read! (%d/%d)\n", num, (int)sizeof(struct mitsu9550_status)); return CUPS_BACKEND_FAILED; } return CUPS_BACKEND_OK; } static const char *mitsu9550_media_types(uint8_t type, uint8_t is_s) { if (is_s) { switch (type & 0xf) { /* values can be 0x0? or 0x4? */ case 0x02: return "CK9015 (4x6)"; case 0x04: return "CK9318 (5x7)"; case 0x05: return "CK9523 (6x9)"; default: return "Unknown"; } return NULL; } switch (type & 0xf) { /* values can be 0x0? or 0x4? */ case 0x01: return "CK9035 (3.5x5)"; case 0x02: return "CK9046 (4x6)"; case 0x03: return "CK9046PST (4x6)"; case 0x04: return "CK9057 (5x7)"; case 0x05: return "CK9069 (6x9)"; case 0x06: return "CK9068 (6x8)"; default: return "Unknown"; } return NULL; } static int validate_media(int type, int media, int cols, int rows) { switch(type) { case P_MITSU_9550: switch(media & 0xf) { case 0x01: /* 3.5x5 */ if (cols != 1812 && rows != 1240) return 1; break; case 0x02: /* 4x6 */ case 0x03: /* 4x6 postcard */ if (cols != 2152) return 1; if (rows != 1416 && rows != 1184 && rows != 1240) return 1; break; case 0x04: /* 5x7 */ if (cols != 1812) return 1; if (rows != 1240 && rows != 2452) return 1; break; case 0x05: /* 6x9 */ if (cols != 2152) return 1; if (rows != 1416 && rows != 2792 && rows != 2956 && rows != 3146) return 1; break; case 0x06: /* V (6x8??) */ if (cols != 2152) return 1; if (rows != 1416 && rows != 2792) return 1; break; default: /* Unknown */ WARNING("Unknown media type %02x\n", media); break; } break; case P_MITSU_9550S: switch(media & 0xf) { case 0x02: /* 4x6 */ case 0x03: /* 4x6 postcard */ if (cols != 2152) return 1; if (rows != 1416 && rows != 1184 && rows != 1240) return 1; break; case 0x04: /* 5x7 */ if (cols != 1812 && rows != 2452) return 1; break; case 0x05: /* 6x9 */ if (cols != 2152) return 1; if (rows != 1416 && rows != 2792 && rows != 2956 && rows != 3146) return 1; break; case 0x06: /* V (6x8??) */ if (cols != 2152) return 1; if (rows != 1416 && rows != 2792) return 1; break; default: /* Unknown */ WARNING("Unknown media type %02x\n", media); break; } break; case P_MITSU_9600: // XXX 9600S doesn't support 5" media at all! switch(media & 0xf) { case 0x01: /* 3.5x5 */ if (cols == 1572) { if (rows == 1076) break; } else if (cols == 3144) { if (rows == 2152) break; } return 1; case 0x02: /* 4x6 */ case 0x03: /* 4x6 postcard */ if (cols == 1868) { if (rows == 1228) break; } else if (cols == 3736) { if (rows == 2458) break; } return 1; case 0x04: /* 5x7 */ if (cols == 1572) { if (rows == 1076 || rows == 2128) break; } else if (cols == 3144) { if (rows == 2152 || rows == 4256) break; } return 1; case 0x05: /* 6x9 */ if (cols == 1868) { if (rows == 1228 || rows == 2442 || rows == 2564 || rows == 2730) break; } else if (cols == 3736) { if (rows == 2458 || rows == 4846 || rows == 5130 || rows == 5462) break; } return 1; case 0x06: /* V (6x8??) */ if (cols == 1868) { if (rows == 1228 || rows == 2442) break; } else if (cols == 3736) { if (rows == 2458 || rows == 4846) break; } return 1; default: /* Unknown */ WARNING("Unknown media type %02x\n", media); break; } break; case P_MITSU_9800: case P_MITSU_9810: // case P_MITSU_9820S: switch(media & 0xf) { case 0x01: /* 3.5x5 */ if (cols != 1572 && rows != 1076) return 1; break; case 0x02: /* 4x6 */ case 0x03: /* 4x6 postcard */ if (cols != 1868 && rows != 1228) return 1; break; case 0x04: /* 5x7 */ if (cols != 1572 && rows != 2128) return 1; break; case 0x05: /* 6x9 */ if (cols != 1868) return 1; if (rows != 1228 && rows != 2442 && rows != 2564 && rows != 2730) return 1; break; case 0x06: /* V (6x8??) */ if (cols != 1868) return 1; if (rows != 1228 && rows != 2442) return 1; break; default: /* Unknown */ WARNING("Unknown media type %02x\n", media); break; } break; case P_MITSU_9800S: switch(media & 0xf) { case 0x02: /* 4x6 */ case 0x03: /* 4x6 postcard */ if (cols != 1868 && rows != 1228) return 1; break; case 0x04: /* 5x7 */ if (cols != 1572 && rows != 2128) return 1; break; case 0x05: /* 6x9 */ if (cols != 1868) return 1; if (rows != 1228 && rows != 2442 && rows != 2564 && rows != 2730) return 1; break; case 0x06: /* V (6x8??) */ if (cols != 1868) return 1; if (rows != 1228 && rows != 2442) return 1; break; default: /* Unknown */ WARNING("Unknown media type %02x\n", media); break; } break; case P_MITSU_CP30D: if (cols != 1600) return 1; if (rows != 1200 && rows != 2100) return 1; // XXX validate media type vs job size. Don't know readback codes. // S == 1600x1200 // L == 1600x2100 ( type 00 ?) break; default: WARNING("Unknown printer type %d\n", type); break; } return CUPS_BACKEND_OK; } static int mitsu9550_main_loop(void *vctx, const void *vjob) { struct mitsu9550_ctx *ctx = vctx; struct mitsu9550_cmd cmd; uint8_t rdbuf[READBACK_LEN]; uint8_t *ptr; int ret; #if 0 int copies = 1; #endif struct mitsu9550_printjob *job = (struct mitsu9550_printjob*) vjob; if (!ctx) return CUPS_BACKEND_FAILED; if (!job) return CUPS_BACKEND_FAILED; /* Okay, let's do this thing */ ptr = job->databuf; int sharpness = job->hdr2.unkc[7]; job->hdr2.unkc[7] = 0; /* Clear "sharpness" parameter */ /* Do the 98xx processing here */ if (!ctx->is_98xx || job->is_raw) goto non_98xx; /* Special CP98xx handling code */ uint8_t *newbuf; uint32_t newlen = 0; int i, remain, planelen; planelen = job->rows * job->cols * 2; remain = (job->hdr1.matte ? 4 : 3) * (planelen + sizeof(struct mitsu9550_plane)) + sizeof(struct mitsu9550_cmd) * (job->hdr1.matte? 2 : 1) + LAMINATE_STRIDE * 2; newbuf = malloc(remain); if (!newbuf) { ERROR("Memory allocation Failure!\n"); return CUPS_BACKEND_RETRY_CURRENT; } DEBUG("Running print data through processing library\n"); /* Create band images for input and output */ struct BandImage input; struct BandImage output; uint8_t *convbuf = malloc(planelen * 3); if (!convbuf) { free(newbuf); ERROR("Memory allocation Failure!\n"); return CUPS_BACKEND_RETRY_CURRENT; } input.origin_rows = input.origin_cols = 0; input.rows = job->rows; input.cols = job->cols; input.imgbuf = job->databuf + sizeof(struct mitsu9550_plane); input.bytes_per_row = job->cols * 3; output.origin_rows = output.origin_cols = 0; output.rows = job->rows; output.cols = job->cols; output.imgbuf = convbuf; output.bytes_per_row = job->cols * 3 * sizeof(uint16_t); if (!ctx->lib.CP98xx_DoConvert(ctx->m98xxdata, &input, &output, job->hdr2.mode, sharpness, job->hdr2.unkc[8])) { free(convbuf); free(newbuf); ERROR("CP98xx_DoConvert() failed!\n"); return CUPS_BACKEND_FAILED; } /* Clear special extension flags used by our backend */ if (job->hdr2.mode == 0x11) job->hdr2.mode = 0x10; job->hdr2.unkc[8] = 0; /* Clear "already reversed" flag */ /* Library is done, but its output is packed YMC16. We need to convert this to planar YMC16, with a header for each plane. */ uint8_t *yPtr, *mPtr, *cPtr; int j, offset; yPtr = newbuf + newlen; memcpy(yPtr, job->databuf, sizeof(struct mitsu9550_plane)); yPtr[3] = 0x10; /* ie 16bpp data */ yPtr += sizeof(struct mitsu9550_plane); newlen += sizeof(struct mitsu9550_plane) + planelen; mPtr = newbuf + newlen; memcpy(mPtr, job->databuf, sizeof(struct mitsu9550_plane)); mPtr[3] = 0x10; /* ie 16bpp data */ mPtr += sizeof(struct mitsu9550_plane); newlen += sizeof(struct mitsu9550_plane) + planelen; cPtr = newbuf + newlen; memcpy(cPtr, job->databuf, sizeof(struct mitsu9550_plane)); cPtr[3] = 0x10; /* ie 16bpp data */ cPtr += sizeof(struct mitsu9550_plane); newlen += sizeof(struct mitsu9550_plane) + planelen; for (offset = 0, i = 0; i < output.rows ; i++) { for (j = 0 ; j < output.cols ; j ++, offset += 3) { yPtr[i*output.cols + j] = convbuf[0 + offset]; mPtr[i*output.cols + j] = convbuf[1 + offset]; cPtr[i*output.cols + j] = convbuf[2 + offset]; } } /* All done with conversion buffer, nuke it */ free(convbuf); /* And finally, append the job footer. */ memcpy(newbuf + newlen, job->databuf + sizeof(struct mitsu9550_plane) + planelen/2 * 3, ctx->footer_len); newlen += sizeof(struct mitsu9550_cmd); /* Clean up, and move pointer to new buffer; */ free(job->databuf); job->databuf = newbuf; job->datalen = newlen; ptr = job->databuf; /* Now handle the matte plane generation */ if (job->hdr1.matte) { if ((i = mitsu98xx_fillmatte(job))) { return i; } } non_98xx: /* Bypass */ if (test_mode >= TEST_MODE_NOPRINT) return CUPS_BACKEND_OK; top: if (ctx->is_s) { int num; /* Send "unknown 1" command */ cmd.cmd[0] = 0x1b; cmd.cmd[1] = 0x53; cmd.cmd[2] = 0xc5; cmd.cmd[3] = 0x9d; if ((ret = send_data(ctx->conn, (uint8_t*) &cmd, sizeof(cmd)))) return CUPS_BACKEND_FAILED; /* Send "unknown query 1" command */ cmd.cmd[0] = 0x1b; cmd.cmd[1] = 0x4b; cmd.cmd[2] = 0x7f; cmd.cmd[3] = 0x00; if ((ret = send_data(ctx->conn, (uint8_t*) &cmd, sizeof(cmd)))) return CUPS_BACKEND_FAILED; ret = read_data(ctx->conn, rdbuf, READBACK_LEN, &num); if (ret < 0) return CUPS_BACKEND_FAILED; // XXX no idea how to interpret this. } if (ctx->conn->type == P_MITSU_9800S) { int num; /* Send "unknown query 2" command */ cmd.cmd[0] = 0x1b; cmd.cmd[1] = 0x4b; cmd.cmd[2] = 0x01; cmd.cmd[3] = 0x00; if ((ret = send_data(ctx->conn, (uint8_t*) &cmd, sizeof(cmd)))) return CUPS_BACKEND_FAILED; ret = read_data(ctx->conn, rdbuf, READBACK_LEN, &num); if (ret < 0) return CUPS_BACKEND_FAILED; // XXX no idea how to interpret this. } /* Sanity-check the printer state. */ { QUERY_STATUS_I; \ if (validate_media(ctx->conn->type, media->type, job->cols, job->rows)) { \ ERROR("Incorrect media (%u) type for printjob (%ux%u)!\n", media->type, job->cols, job->rows); \ return CUPS_BACKEND_HOLD; \ } \ QUERY_STATUS_II; \ QUERY_STATUS_IV; \ } /*** Now it's time for the actual print job! ***/ /* Send printjob headers from spool data */ if (job->hdr1_present) if ((ret = send_data(ctx->conn, (uint8_t*) &job->hdr1, sizeof(job->hdr1)))) return CUPS_BACKEND_FAILED; if (job->hdr2_present) if ((ret = send_data(ctx->conn, (uint8_t*) &job->hdr2, sizeof(job->hdr2)))) return CUPS_BACKEND_FAILED; if (job->hdr3_present) if ((ret = send_data(ctx->conn, (uint8_t*) &job->hdr3, sizeof(job->hdr3)))) return CUPS_BACKEND_FAILED; if (job->hdr4_present) if ((ret = send_data(ctx->conn, (uint8_t*) &job->hdr4, sizeof(struct mitsu9550_hdr4)))) return CUPS_BACKEND_FAILED; if (ctx->is_s) { /* I think this a "clear memory' command...? */ cmd.cmd[0] = 0x1b; cmd.cmd[1] = 0x5a; cmd.cmd[2] = 0x43; cmd.cmd[3] = 0x00; if ((ret = send_data(ctx->conn, (uint8_t*) &cmd, sizeof(cmd)))) return CUPS_BACKEND_FAILED; } /* Send over plane data */ while(ptr < (job->databuf + job->datalen)) { struct mitsu9550_plane *plane = (struct mitsu9550_plane *)ptr; if (plane->cmd[0] != 0x1b || plane->cmd[1] != 0x5a || plane->cmd[2] != 0x54) break; planelen = be16_to_cpu(plane->rows) * be16_to_cpu(plane->cols); if (plane->cmd[3] == 0x10) planelen *= 2; if ((ret = send_data(ctx->conn, (uint8_t*) ptr, sizeof(struct mitsu9550_plane)))) return CUPS_BACKEND_FAILED; ptr += sizeof(struct mitsu9550_plane); if ((ret = send_data(ctx->conn, (uint8_t*) ptr, planelen))) return CUPS_BACKEND_FAILED; ptr += planelen; } /* Query statuses after sending data */ { QUERY_STATUS_I; QUERY_STATUS_II; QUERY_STATUS_IV; } /* Send "end data" command */ if (ctx->conn->type == P_MITSU_9550S) { /* Override spool, which may be wrong */ cmd.cmd[0] = 0x1b; cmd.cmd[1] = 0x50; cmd.cmd[2] = 0x47; cmd.cmd[3] = 0x00; if ((ret = send_data(ctx->conn, (uint8_t*) &cmd, sizeof(cmd)))) return CUPS_BACKEND_FAILED; } else if (ctx->conn->type == P_MITSU_9800S) { /* Override spool, which may be wrong */ cmd.cmd[0] = 0x1b; cmd.cmd[1] = 0x50; cmd.cmd[2] = 0x4e; cmd.cmd[3] = 0x00; if ((ret = send_data(ctx->conn, (uint8_t*) &cmd, sizeof(cmd)))) return CUPS_BACKEND_FAILED; } else { /* Send from spool file */ if ((ret = send_data(ctx->conn, ptr, ctx->footer_len))) return CUPS_BACKEND_FAILED; } ptr += ctx->footer_len; /* Don't forget the 9810's matte plane */ if (job->hdr1.matte) { struct mitsu9550_plane *plane = (struct mitsu9550_plane *)ptr; planelen = be16_to_cpu(plane->rows) * be16_to_cpu(plane->cols); if (plane->cmd[3] == 0x10) planelen *= 2; // XXX include a status loop here too? if ((ret = send_data(ctx->conn, (uint8_t*) ptr, sizeof(struct mitsu9550_plane)))) return CUPS_BACKEND_FAILED; ptr += sizeof(struct mitsu9550_plane); if ((ret = send_data(ctx->conn, (uint8_t*) ptr, planelen))) return CUPS_BACKEND_FAILED; ptr += planelen; /* Send "lamination end data" command from spool file */ if ((ret = send_data(ctx->conn, ptr, sizeof(cmd)))) return CUPS_BACKEND_FAILED; // ptr += sizeof(cmd); /* Unnecessary */ } /* Status loop, run until printer reports completion */ while(1) { sleep(1); QUERY_STATUS_I; QUERY_STATUS_II; if (ctx->conn->type == P_MITSU_CP30D) { ret = mitsu9550_get_status(ctx, rdbuf, CP9XXX_STS_x32); if (ret < 0) return CUPS_BACKEND_FAILED; if (sts30->err == CP30_ERR_TRAYFULL) { ERROR("Output Tray Full!\n"); return CUPS_BACKEND_STOP; } // XXX figure out remaining copy count? // print copy remaining if (sts30->sts == CP30_STS_IDLE) /* If printer transitions to idle */ break; // XXX if (fast_return && copies_remaining == 0) break... if (fast_return && sts30->sts != CP30_STS_IDLE) { INFO("Fast return mode enabled.\n"); break; } QUERY_STATUS_IIIB; } else { ret = mitsu9550_get_status(ctx, rdbuf, CP9XXX_STS_x30); if (ret < 0) return CUPS_BACKEND_FAILED; INFO("%03d copies remaining\n", be16_to_cpu(sts->copies)); if (!sts->sts1) /* If printer transitions to idle */ break; if (fast_return && !be16_to_cpu(sts->copies)) { /* No remaining prints */ INFO("Fast return mode enabled.\n"); break; } if (fast_return && !sts->sts5) { INFO("Fast return mode enabled.\n"); break; } QUERY_STATUS_III; } } INFO("Print complete\n"); return CUPS_BACKEND_OK; } static void mitsu9550_dump_media(struct mitsu9550_ctx *ctx, struct mitsu9550_media *resp) { INFO("Media type : %02x (%s)\n", resp->type, (ctx->conn->type == P_MITSU_CP30D ? cp30_media_types(be16_to_cpu(resp->max)): mitsu9550_media_types(resp->type, ctx->is_s))); INFO("Media remaining : %03d/%03d\n", (ctx->conn->type == P_MITSU_CP30D) ? be16_to_cpu(resp->remain2) : be16_to_cpu(resp->remain), be16_to_cpu(resp->max)); } static void mitsu9550_dump_status(struct mitsu9550_status *resp) { INFO("Printer status : %02x (%s)\n", resp->sts1, resp->sts1 ? "Printing": "Idle"); INFO("Pages remaining : %03d\n", be16_to_cpu(resp->copies)); INFO("Other status : %02x %02x %02x %02x %02x %02x\n", resp->sts2, resp->sts3, resp->sts4, resp->sts5, resp->sts6, resp->sts7); } static void mitsucp30_dump_status(struct mitsucp30_status *resp) { INFO("Printer status : %02x %02x\n", resp->sts, resp->sts2); INFO("Printer error : %s (%04x)\n", cp30_errors(be16_to_cpu(resp->err)), be16_to_cpu(resp->err)); } static void mitsu9550_dump_status2(struct mitsu9550_status2 *resp) { INFO("Prints remaining on media : %03d\n", be16_to_cpu(resp->remain)); } static int mitsu9550_query_media(struct mitsu9550_ctx *ctx) { struct mitsu9550_media resp; int ret; ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_MEDIA); if (!ret) mitsu9550_dump_media(ctx, &resp); return ret; } static int mitsu9550_query_status2(struct mitsu9550_ctx *ctx) { struct mitsu9550_status2 resp; int ret; ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_x21); if (!ret && ctx->conn->type != P_MITSU_CP30D) mitsu9550_dump_status2(&resp); return ret; } static int mitsu9550_query_status(struct mitsu9550_ctx *ctx) { int ret; if (ctx->conn->type == P_MITSU_CP30D) { struct mitsucp30_status resp; ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_x32); if (!ret) { mitsucp30_dump_status(&resp); ret = mitsu9550_query_status2(ctx); } } else { struct mitsu9550_status resp; ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_x30); if (!ret) { mitsu9550_dump_status(&resp); ret = mitsu9550_query_status2(ctx); } } return ret; } static int mitsu9550_query_statusX(struct mitsu9550_ctx *ctx) { struct mitsu9550_status2 resp; int ret; #if 0 int i; for (i = 0 ; i < 256 ; i++) { ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, i); if (!ret) { DEBUG("Query %02x OK\n", i); } } #else ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_x20); ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_FWVER); ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_x22); ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_x26); ret = mitsu9550_get_status(ctx, (uint8_t*) &resp, CP9XXX_STS_x32); #endif return ret; } static int mitsu9550_query_serno(struct dyesub_connection *conn, char *buf, int buf_len) { struct mitsu9550_cmd cmd; uint8_t rdbuf[READBACK_LEN]; uint8_t *ptr; int ret, num, i; cmd.cmd[0] = 0x1b; cmd.cmd[1] = 0x72; cmd.cmd[2] = 0x6e; cmd.cmd[3] = 0x00; if ((ret = send_data(conn, (uint8_t*) &cmd, sizeof(cmd)))) return (ret < 0) ? ret : CUPS_BACKEND_FAILED; ret = read_data(conn, rdbuf, READBACK_LEN, &num); if (ret < 0) return CUPS_BACKEND_FAILED; if ((unsigned int)num < sizeof(cmd) + 1) /* Short read */ return CUPS_BACKEND_FAILED; if (rdbuf[0] != 0xe4 || rdbuf[1] != 0x72 || rdbuf[2] != 0x6e || rdbuf[3] != 0x00) /* Bad response */ return CUPS_BACKEND_FAILED; /* If response is truncated, handle it */ num -= (sizeof(cmd) + 1); if ((unsigned int) num != rdbuf[4]) WARNING("Short serno read! (%d vs %u)\r\n", num, rdbuf[4]); /* model and serial number are encoded as 16-bit unicode, little endian, separated by spaces. */ i = num; ptr = rdbuf + 5; while (i > 0 && buf_len > 1) { if (*ptr != 0x20) *buf++ = *ptr; buf_len--; ptr += 2; i -= 2; } *buf = 0; /* Null-terminate the returned string */ return ret; } static int mitsu9550_cancel_job(struct mitsu9550_ctx *ctx) { int ret; uint8_t buf[2] = { 0x1b, 0x44 }; ret = send_data(ctx->conn, buf, sizeof(buf)); return ret; } static void mitsu9550_cmdline(void) { DEBUG("\t\t[ -m ] # Query media\n"); DEBUG("\t\t[ -s ] # Query status\n"); DEBUG("\t\t[ -X ] # Cancel current job\n"); // DEBUG("\t\t[ -Z ] # Dump all parameters\n"); } static int mitsu9550_cmdline_arg(void *vctx, int argc, char **argv) { struct mitsu9550_ctx *ctx = vctx; int i, j = 0; if (!ctx) return -1; while ((i = getopt(argc, argv, GETOPT_LIST_GLOBAL "msXZ")) >= 0) { switch(i) { GETOPT_PROCESS_GLOBAL case 'm': j = mitsu9550_query_media(ctx); break; case 's': j = mitsu9550_query_status(ctx); INFO("Firmware Version: %s\n", ctx->fwver); break; case 'X': j = mitsu9550_cancel_job(ctx); break; case 'Z': j = mitsu9550_query_statusX(ctx); break; default: break; /* Ignore completely */ } if (j) return j; } return CUPS_BACKEND_OK; } static int mitsu9550_query_markers(void *vctx, struct marker **markers, int *count) { struct mitsu9550_ctx *ctx = vctx; struct mitsu9550_media media; /* Query printer status */ if (mitsu9550_get_status(ctx, (uint8_t*) &media, CP9XXX_STS_MEDIA)) return CUPS_BACKEND_FAILED; if (ctx->conn->type == P_MITSU_CP30D) { ctx->marker.levelnow = be16_to_cpu(media.remain2); } else { ctx->marker.levelnow = be16_to_cpu(media.remain); } *markers = &ctx->marker; *count = 1; return CUPS_BACKEND_OK; } static const char *mitsu9550_prefixes[] = { "mitsu9xxx", // Family driver, do not nuke. // Backwards compatibility "mitsu9000", "mitsu9500", "mitsu9550", "mitsu9600", "mitsu9800", "mitsu9810", NULL }; /* Exported */ const struct dyesub_backend mitsu9550_backend = { .name = "Mitsubishi CP9xxx family", .version = "0.61" " (lib " LIBMITSU_VER ")", .uri_prefixes = mitsu9550_prefixes, .cmdline_usage = mitsu9550_cmdline, .cmdline_arg = mitsu9550_cmdline_arg, .init = mitsu9550_init, .attach = mitsu9550_attach, .teardown = mitsu9550_teardown, .cleanup_job = mitsu9550_cleanup_job, .read_parse = mitsu9550_read_parse, .main_loop = mitsu9550_main_loop, .query_serno = mitsu9550_query_serno, .query_markers = mitsu9550_query_markers, .devices = { { 0x06d3, 0x0395, P_MITSU_9550, NULL, "mitsubishi-9000dw"}, // XXX -am instead? { 0x06d3, 0x0394, P_MITSU_9550, NULL, "mitsubishi-9000dw"}, { 0x06d3, 0x0393, P_MITSU_9550, NULL, "mitsubishi-9500dw"}, { 0x06d3, 0x03a1, P_MITSU_9550, NULL, "mitsubishi-9550dw"}, { 0x06d3, 0x03a1, P_MITSU_9550, NULL, "mitsubishi-9550d"}, /* Duplicate */ { 0x06d3, 0x03a5, P_MITSU_9550S, NULL, "mitsubishi-9550dw-s"}, // or DZ/DZS/DZU { 0x06d3, 0x03a5, P_MITSU_9550S, NULL, "mitsubishi-9550dz"}, /* Duplicate */ { 0x06d3, 0x03a9, P_MITSU_9600, NULL, "mitsubishi-9600dw"}, // { 0x06d3, USB_PID_MITSU_9600D, P_MITSU_9600S, NULL, "mitsubishi-9600dw-s"}, { 0x06d3, 0x03ab, P_MITSU_CP30D, NULL, "mitsubishi-cp30dw"}, { 0x06d3, 0x03ad, P_MITSU_9800, NULL, "mitsubishi-9800dw"}, { 0x06d3, 0x03ad, P_MITSU_9800, NULL, "mitsubishi-9800d"}, /* Duplicate */ { 0x06d3, 0x03ae, P_MITSU_9800S, NULL, "mitsubishi-9800dw-s"}, { 0x06d3, 0x03ae, P_MITSU_9800S, NULL, "mitsubishi-9800dz"}, /* Duplicate */ { 0x06d3, 0x3b21, P_MITSU_9810, NULL, "mitsubishi-9810dw"}, { 0x06d3, 0x3b21, P_MITSU_9810, NULL, "mitsubishi-9810d"}, /* Duplicate */ // { 0x06d3, USB_PID_MITSU_9820DS, P_MITSU_9820S, NULL, "mitsubishi-9820dw-s"}, { 0, 0, 0, NULL, NULL} } }; /* Mitsubish CP-30/9500/9550/9600/9800/9810/9820 spool format: Spool file consists of 3 or 4 50-byte headers, followed by three image planes, each with a 12-byte header, then a 4 or 6-byte footer. All multi-byte numbers are big endian. ~~~ Header 1 1b 57 20 2e 00 QQ QQ 00 00 00 00 00 00 00 XX XX :: XX XX == columns YY YY 00 00 00 00 00 00 00 00 00 00 00 00 00 00 :: YY YY == rows 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 :: QQ == 0x0a90 on 9810, 0x0a10 on all others. 00 00 ~~~ Header 2 1b 57 21 2e 00 GG 00 HH QQ QQ 00 00 00 00 00 00 :: ZZ ZZ = num copies (>= 0x01) 00 00 00 00 00 00 00 00 00 00 00 00 ZZ ZZ 00 00 :: YY = 00/80 Fine/SuperFine (9550), 10/80 Fine/Superfine (98x0), 00 (9600), 0x80/0x00 Powersave/Normal (CP30) XX 00 00 00 00 00 YY 00 00 00 00 00 00 00 SS TT :: XX = 00 normal, 83 Cut 2x6 (9550 only!) RR II :: QQ QQ = 0x0803 on 9550, 0x0801 on 98x0, 0x0003 on 9600, 0xa803 on 9500, 0x0802 on CP30 :: RR = 01 for "use LUT" on 98xx, 0x00 otherwise. Extension to stock. :: TT = 01 for "already reversed". Extension to stock. :: SS == sharpening level, 0 for off, 1-10 otherwise. Extesion to stock. :: GG == 0x00 on CP30, 0x80 on others :: HH == 0x20 on CP30, 0x22 on others :: II == 0x00 on CP30, 0x01 on others. ~~~ Header 3 (9550, 9800-S, and CP30 only..) 1b 57 22 2e 00 QQ 00 00 00 00 00 XX 00 00 00 00 :: XX = 00 normal, 01 FineDeep 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 :: QQ = 0xf0 on 9500, 0x40 on the rest 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ~~~ Header 4 (all but 9550-S and 9800-S, involves error policy?) 1b 57 26 2e 00 QQ TT 00 00 00 00 SS RR ZZ VV WW :: QQ = 0x70 on 9550/98x0, 0x60 on 9600 or 9800S, 0x3f on CP30 [also seen 0x20 & 0x00 on 9550S] WW 00 WW 00 00 00 00 00 00 00 00 00 00 00 00 00 :: RR = 0x01 on 9550/98x0/CP30, 0x00 on 9600 [ "ignore errors? ] 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 :: SS = 0x01 on 9800S, 0x00 otherwise. 00 00 :: ZZ = Unknown; 0x01 [9550S/CP30] & 0x00 [9500]. :: TT = 0x80 on CP30, 0x00 otherwise :: VV = 0x80 on CP30, 0x00 otherwise :: WW = 0x10 on CP30, 0x00 otherwise ~~~~ Data follows: Format is: planar YMC16 for 98x0 (but only 12 bits used, BIG endian) planar RGB for all others 1b 5a 54 ?? RR RR CC CC 07 14 04 d8 :: 0714 == columns, 04d8 == rows :: RRRR == row offset for data, CCCC == col offset for data :: ?? == 0x00 for 8bpp, 0x10 for 16/12bpp. :: 0x80 for PACKED BGR! Data follows immediately, no padding. 1b 5a 54 00 00 00 00 00 07 14 04 d8 :: Another plane. Data follows immediately, no padding. 1b 5a 54 00 00 00 00 00 07 14 04 d8 :: Another plane. Data follows immediately, no padding. ~~~~ Footer: 1b 50 41 00 (9500AM) 1b 50 46 00 (9550) 1b 50 47 00 (9550-S) 1b 50 48 00 (9600) 1b 50 4c 00 (9800/9810) 1b 50 4d 00 (9000) 1b 50 4e 00 (9800-S) 1b 50 51 00 (CP3020DA) 1b 50 57 00 (9500) 1b 50 52 00 00 00 (CP30) Unknown: 9600-S, 9820-S, 1 other.. ~~~~ Lamination data follows (on 9810 only, if matte selected) 1b 5a 54 10 00 00 00 00 06 24 04 34 Data follows immediately, no padding. 1b 50 56 00 (Lamination footer) ~~~~ QUESTIONS: * Lamination control? * Other 9550 multi-cut modes (on 6x9 media: 4x6*2, 4.4x6*2, 3x6*3, 2x6*4) * 9600/98x0 multi-cut modes? *********************************************************************** * Mitsubishi CP-9xxx Communications Protocol: JOB CANCEL -> 1b 44 [[ Unknown query 1 ]] -> 1b 4b 7f 00 <- eb 4b 8f 00 02 00 5e [[ '02' seems to be a length ]] [[ unknown query 2, 9800-only ]] -> 1b 4b 01 00 <- e4 4b 01 00 02 00 78 PRINT START -> 1b 50 47 00 [9550S] -> 1b 50 4e 00 [9800S] -> 1b 50 56 00 [9810 Lamination] [[ see "footer" above for other models ]] [[ Unknown 1 ]] -> 1b 51 c5 9d [[ Unknown ]] -> 1b 53 00 00 [[ Unknown ]] -> 1b 53 c5 9d [[ Status Query B ]] -> 1b 56 20 00 [ CP30 ] <- 20 2e 00 0a 10 00 00 00 00 00 00 00 CC CC RR RR :: CC == cols 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 :: RR == rows (04b0 for L??) 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 [[ Status Query C ]] -> 1b 56 21 00 [ Most models ] <- 21 2e 00 80 00 22 a8 0b 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 QQ 00 00 00 00 :: QQ == Prints in job? 00 00 00 00 00 00 00 00 00 00 NN NN 0A 00 00 01 :: NN NN = Remaining media 21 2e 00 00 00 20 08 02 00 00 00 00 00 00 00 00 [ CP30 ] 00 00 00 00 00 00 00 00 00 00 00 XX 00 00 00 00 :: XX == seen 01 and 02. Unknown. 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 [[ Status Query D (unknown, possibly lifetime print count?) ]] -> 1b 56 22 00 [ CP30 ] <- 22 2e 00 40 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 FIRWARE VERSIONS -> 1b 56 23 00 [ CP30 ] <- 23 2e 00 00 00 00 32 32 33 46 31 30 32 32 34 42 223F10 224B10 ... 222A10 31 30 00 00 00 00 00 00 32 32 32 41 31 30 00 00 00 00 00 05 d9 3f 79 20 00 00 13 97 00 00 00 00 MEDIA INFO -> 1b 56 24 00 <- 24 2e 00 00 00 00 00 00 00 00 00 00 XX 00 TT 00 :: TT = Type (!CP30) ; XX = 0x02 on CP30, 0x00 otherwise. 00 00 00 00 00 00 00 00 00 00 00 00 MM MM N1 N1 :: MM MM = Max prints NN NN 00 00 00 00 00 00 00 00 00 00 00 00 00 00 :: NN NN = Remaining (!CP30) ; N1 N1 = Remaining (CP30) [[ Status Query E ]] -> 1b 56 26 00 [ CP30 ] <- 26 2e 00 3f 80 00 00 00 00 00 01 01 80 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 Status Query [All models except CP30] -> 1b 56 30 00 -> 30 2e 00 00 00 00 MM 00 NN NN ZZ 00 00 00 00 00 :: MM, NN, ZZ QQ RR SS 00 00 00 00 00 00 00 00 00 00 00 00 00 :: QQ, RR, SS 00 00 00 00 00 00 00 00 00 00 00 00 TT UU 00 00 :: TT, UU Status Query [CP30] -> 1b 56 32 00 <- 32 2e 00 00 00 43 00 50 00 33 00 30 00 44 00 20 :: Unicode, "CP30D 204578" 00 32 00 30 00 34 00 35 00 37 00 38 00 00 SS SS :: SS SS = status code (see CP30_STS_*) 00 EE EE 00 00 00 NN 00 00 00 00 00 00 00 00 00 :: NN = remaining prints on paper :: EE EE = error code (see CP30_ERR_*) Status Query X (unknown) -> 1b 56 33 00 <- ??? 48 bytes? Status Query Y (unknown) -> 1b 56 36 00 <- ??? 48 bytes? [[ Header 1 -- See above ]] -> 1b 57 20 2e .... [[ Header 2 -- See above ]] -> 1b 57 21 2e .... [[ Header 3 -- See above ]] -> 1b 57 22 2e .... [[ Header 4 -- See above ]] -> 1b 57 26 2e .... DATA Start [[ Or maybe it's "DATA Clear" ? Only seen on -S models ]] -> 1b 5a 43 00 PLANE Data -> 1b 5a 54 ?? 00 00 00 00 XX XX YY YY :: XX XX == Columns, YY YY == Rows :: ?? == x00 8bpp, x10 16bpp Followed by image plane data, XXXX * YYYY [ * 2 ] bytes [ Three planes are needed! ] Query Model & FW Version (XXX Confirm this!) -> 1b 72 01 00 <- e4 82 01 00 LL 39 00 35 00 35 00 30 00 5a 00 20 00 41 00 32 00 30 00 30 00 36 00 37 00 'LL' is length. Data is returned in 16-bit unicode, LE. Contents are model ('9550Z'), then space, then serialnum ('A20067') [[ Unknown query.. "Printer number" related? Seen in driver dump ]] -> 1b 72 10 00 <- e4 82 10 00 LL [ 10 unknown bytes. guess. ] Query Model & Serial number -> 1b 72 6e 00 <- e4 82 6e 00 LL 39 00 35 00 35 00 30 00 5a 00 20 00 41 00 32 00 30 00 30 00 36 00 37 00 'LL' is length. Data is returned in 16-bit unicode, LE. Contents are model ('9550Z'), then space, then serialnum ('A20067') **** After print starts, loop status/status b/media queries until printer idle MM, QQ RR SS, TT UU <- 00 3e 00 00 8a 44 :: Idle. 00 7e 00 00 8a 44 :: Plane data submitted, pre "end data" cmd 00 7e 40 01 8a 44 :: "end data" sent 30 7e 40 01 8a 44 38 7e 40 01 8a 44 59 7e 40 01 8a 44 59 7e 40 00 8a 44 4d 7e 40 00 8a 44 [...] 43 7e 40 00 82 44 [...] 50 7e 40 00 80 44 [...] 31 7e 40 00 7d 44 [...] 00 3e 00 00 80 44 :: Idle. Also seen: 00 3e 00 00 96 4b :: Idle 00 be 00 00 96 4b :: Data submitted, pre "start" 00 be 80 01 96 4b :: print start sent 30 be 80 01 96 4c [...] 30 be 80 01 89 4b 38 be 80 01 8a 4b 59 be 80 01 8b 4b [...] 4d be 80 01 89 4b [...] 43 be 80 01 89 4b [...] 50 be 80 01 82 4b [...] 31 be 80 01 80 4b [...] Working theory of interpreting the status flags: MM :: 00 is idle, else mechanical printer state. NN :: Remaining prints in job, or 0x00 0x00 when idle. QQ :: ?? 0x3e + 0x40 or 0x80 (see below) RR :: ?? 0x00 is idle, 0x40 or 0x80 is "printing"? SS :: ?? 0x00 means "ready for another print" but 0x01 is "busy" TT :: ?? seen values between 0x7c through 0x96) UU :: ?? seen values between 0x43 and 0x4c -- temperature? ZZ :: ?? Error code (08 = Door open on 9600) *** Other printer commands seen: [[ Set error policy ?? aka "header 4" ]] -> 1b 57 26 2e 00 QQ 00 00 00 00 00 00 RR ZZ 00 00 :: QQ/RR/ZZ 00 00 00 [9550S] 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 :: 20 01 00 [9550S w/ ignore failures on] 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 :: 70 01 01 [9550] 00 00 */