Fix PT-P710BT raster protocol handling and add regression coverage.
Align print command payload generation with model-specific protocol requirements, add verbose CLI diagnostics and robust status retries, and introduce protocol regression tests to prevent future GF/ESC i z regressions. Made-with: Cursor
This commit is contained in:
@@ -24,6 +24,7 @@ static void usage(const char *argv0)
|
||||
" -t, --threshold N しきい値 0–255(既定 %u、PNG/SVG)\n"
|
||||
" --trim-right[=DOTS] 右側空白を削減(省略時は左余白、失敗時 0)\n"
|
||||
" -n, --dry-run 読み込みと check_raster のみ(USB なし)\n"
|
||||
" -v, --verbose 印刷前情報と印刷後ステータスを標準出力\n"
|
||||
" -p, --pid HEX USB 製品 ID(既定: P900W の 0x2085)。例: P750W 0x2062、P710BT 0x20af\n"
|
||||
" -S, --status ステータスを表示して終了\n"
|
||||
" -V, --version バージョンを表示して終了\n"
|
||||
@@ -101,6 +102,53 @@ static int read_file(const char *path, uint8_t **out, size_t *out_len)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void verbose_print_pre_print_info(libptouch_ctx *ctx,
|
||||
const libptouch_raster_params_t *params,
|
||||
size_t data_len)
|
||||
{
|
||||
if (!ctx || !params)
|
||||
return;
|
||||
|
||||
printf("=== verbose: pre-print info ===\n");
|
||||
printf("raster bytes: %zu\n", data_len);
|
||||
printf("raster size: %ux%u dots (length x width)\n",
|
||||
(unsigned)params->width_dots, (unsigned)params->height_dots);
|
||||
printf("margin_mm: %u\n", (unsigned)params->margin_mm);
|
||||
|
||||
libptouch_media_info_t mi;
|
||||
if (libptouch_get_current_media_info(ctx, &mi) == LIBPTOUCH_OK) {
|
||||
printf("media width code: 0x%02X\n", (unsigned)mi.media_width_code);
|
||||
printf("media kind code: 0x%02X\n", (unsigned)mi.media_kind_code);
|
||||
printf("tape width: %.1f mm\n", mi.tape_width_mm);
|
||||
printf("printable dots: %u\n", (unsigned)mi.printable_dots);
|
||||
printf("left/right margins: %u/%u dots\n",
|
||||
(unsigned)mi.left_margin_dots, (unsigned)mi.right_margin_dots);
|
||||
printf("print/feed dpi: %.1f/%.1f\n", mi.print_dpi, mi.feed_dpi);
|
||||
} else {
|
||||
printf("media info: unavailable (%s)\n", libptouch_strerror(ctx));
|
||||
}
|
||||
|
||||
uint8_t st[LIBPTOUCH_STATUS_LENGTH];
|
||||
if (libptouch_get_status(ctx, st) == LIBPTOUCH_OK) {
|
||||
libptouch_status_fprint(stdout, st);
|
||||
} else {
|
||||
printf("status: unavailable (%s)\n", libptouch_strerror(ctx));
|
||||
}
|
||||
}
|
||||
|
||||
static void verbose_print_post_print_status(libptouch_ctx *ctx, const char *label)
|
||||
{
|
||||
if (!ctx)
|
||||
return;
|
||||
printf("=== verbose: post-print status (%s) ===\n", label);
|
||||
uint8_t st[LIBPTOUCH_STATUS_LENGTH];
|
||||
if (libptouch_get_status(ctx, st) == LIBPTOUCH_OK) {
|
||||
libptouch_status_fprint(stdout, st);
|
||||
} else {
|
||||
printf("status: unavailable (%s)\n", libptouch_strerror(ctx));
|
||||
}
|
||||
}
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
const char *file = NULL;
|
||||
@@ -108,6 +156,7 @@ int main(int argc, char **argv)
|
||||
unsigned threshold = LIBPTOUCH_PNG_DEFAULT_THRESHOLD;
|
||||
unsigned usb_pid_arg = 0;
|
||||
int dry_run = 0;
|
||||
int verbose = 0;
|
||||
int trim_right_enabled = 0;
|
||||
int trim_right_auto = 0;
|
||||
unsigned trim_right_dots = 0;
|
||||
@@ -120,6 +169,7 @@ int main(int argc, char **argv)
|
||||
{ "threshold", required_argument, NULL, 't' },
|
||||
{ "trim-right", optional_argument, NULL, 'r' },
|
||||
{ "dry-run", no_argument, NULL, 'n' },
|
||||
{ "verbose", no_argument, NULL, 'v' },
|
||||
{ "pid", required_argument, NULL, 'p' },
|
||||
{ "status", no_argument, NULL, 'S' },
|
||||
{ "version", no_argument, NULL, 'V' },
|
||||
@@ -128,7 +178,7 @@ int main(int argc, char **argv)
|
||||
};
|
||||
|
||||
int c;
|
||||
while ((c = getopt_long(argc, argv, "w:H:f:t:r::p:nhSV", longopts, NULL)) !=
|
||||
while ((c = getopt_long(argc, argv, "w:H:f:t:r::p:nvhSV", longopts, NULL)) !=
|
||||
-1) {
|
||||
switch (c) {
|
||||
case 'w':
|
||||
@@ -151,6 +201,9 @@ int main(int argc, char **argv)
|
||||
case 'n':
|
||||
dry_run = 1;
|
||||
break;
|
||||
case 'v':
|
||||
verbose = 1;
|
||||
break;
|
||||
case 'r':
|
||||
trim_right_enabled = 1;
|
||||
if (!optarg) {
|
||||
@@ -397,8 +450,13 @@ int main(int argc, char **argv)
|
||||
}
|
||||
}
|
||||
|
||||
if (verbose)
|
||||
verbose_print_pre_print_info(ctx, ¶ms, data_len);
|
||||
|
||||
e = libptouch_print_raster(ctx, data, data_len, ¶ms);
|
||||
if (e != LIBPTOUCH_OK) {
|
||||
if (verbose)
|
||||
verbose_print_post_print_status(ctx, "print failed");
|
||||
fprintf(stderr, "print_raster: %s\n",
|
||||
libptouch_strerror(ctx));
|
||||
libptouch_close(ctx);
|
||||
@@ -410,6 +468,9 @@ int main(int argc, char **argv)
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (verbose)
|
||||
verbose_print_post_print_status(ctx, "print success");
|
||||
|
||||
libptouch_close(ctx);
|
||||
libptouch_destroy(ctx);
|
||||
if (data_from_lib)
|
||||
|
||||
@@ -7,6 +7,7 @@
|
||||
|
||||
#include "libptouch_internal.h"
|
||||
#include "libptouch_layout.h"
|
||||
#include "libptouch_protocol.h"
|
||||
|
||||
#include <libusb.h>
|
||||
#include <stdio.h>
|
||||
@@ -102,7 +103,7 @@ libptouch_err_t libptouch_print_raster(libptouch_ctx *ctx,
|
||||
(void)right_dots;
|
||||
|
||||
unsigned head_dots = prof->head_width_dots;
|
||||
size_t line_payload = (size_t)((head_dots + 7u) / 8u);
|
||||
size_t line_payload = ptouch_line_payload_bytes(head_dots);
|
||||
size_t gf_packet = 3u + line_payload;
|
||||
|
||||
uint32_t wd = params->width_dots;
|
||||
@@ -139,21 +140,6 @@ libptouch_err_t libptouch_print_raster(libptouch_ctx *ctx,
|
||||
}
|
||||
|
||||
uint32_t lines = ht;
|
||||
uint8_t n5 = (uint8_t)(lines & 0xFFu);
|
||||
uint8_t n6 = (uint8_t)((lines >> 8) & 0xFFu);
|
||||
uint8_t n7 = (uint8_t)((lines >> 16) & 0xFFu);
|
||||
uint8_t n8 = (uint8_t)((lines >> 24) & 0xFFu);
|
||||
|
||||
uint8_t n2_paper = 0x09u;
|
||||
if (media_kind == 0x03u)
|
||||
n2_paper = 0x00u;
|
||||
else if (media_kind == 0x11u)
|
||||
n2_paper = 0x11u;
|
||||
else if (media_kind == 0x17u)
|
||||
n2_paper = 0x17u;
|
||||
else if (media_kind == 0x13u)
|
||||
n2_paper = 0x13u;
|
||||
|
||||
uint8_t head[256];
|
||||
size_t pos = 0;
|
||||
memset(head + pos, 0, 200);
|
||||
@@ -165,18 +151,28 @@ libptouch_err_t libptouch_print_raster(libptouch_ctx *ctx,
|
||||
memcpy(head + pos, raster_mode, sizeof(raster_mode));
|
||||
pos += sizeof(raster_mode);
|
||||
|
||||
uint8_t esc_iz[] = { 0x1B, 0x69, 0x7A, 0x0Eu, n2_paper, media_w,
|
||||
0x00u, n5, n6, n7, n8, 0x02u, 0x00u };
|
||||
uint8_t esc_iz[13];
|
||||
ptouch_fill_esc_iz(esc_iz, media_kind, media_w, lines);
|
||||
memcpy(head + pos, esc_iz, sizeof(esc_iz));
|
||||
pos += sizeof(esc_iz);
|
||||
|
||||
static const uint8_t esc_im[] = { 0x1B, 0x69, 0x4D, 0x40 };
|
||||
memcpy(head + pos, esc_im, sizeof(esc_im));
|
||||
pos += sizeof(esc_im);
|
||||
static const uint8_t esc_ia[] = { 0x1B, 0x69, 0x41, 0x01 };
|
||||
memcpy(head + pos, esc_ia, sizeof(esc_ia));
|
||||
pos += sizeof(esc_ia);
|
||||
static const uint8_t esc_ik[] = { 0x1B, 0x69, 0x4B, 0x0C };
|
||||
|
||||
/*
|
||||
* ESC i A ("cut each n labels") is not supported on PT-P710BT class
|
||||
* devices. Sending it can trigger a communication error (red LED blink).
|
||||
* Keep it only for 560-dot family.
|
||||
*/
|
||||
if (prof->head_width_dots > 128u) {
|
||||
static const uint8_t esc_ia[] = { 0x1B, 0x69, 0x41, 0x01 };
|
||||
memcpy(head + pos, esc_ia, sizeof(esc_ia));
|
||||
pos += sizeof(esc_ia);
|
||||
}
|
||||
|
||||
/* Chain printing off; safer default for 128-dot family as well. */
|
||||
static const uint8_t esc_ik[] = { 0x1B, 0x69, 0x4B, 0x08 };
|
||||
memcpy(head + pos, esc_ik, sizeof(esc_ik));
|
||||
pos += sizeof(esc_ik);
|
||||
|
||||
@@ -199,7 +195,6 @@ libptouch_err_t libptouch_print_raster(libptouch_ctx *ctx,
|
||||
}
|
||||
|
||||
size_t row_b = ((size_t)wd + 7u) / 8u;
|
||||
static const uint8_t g_hdr[] = { 0x47, 0x46, 0x00 };
|
||||
|
||||
uint8_t *gbuf = (uint8_t *)malloc(gf_packet);
|
||||
if (!gbuf) {
|
||||
@@ -213,7 +208,7 @@ libptouch_err_t libptouch_print_raster(libptouch_ctx *ctx,
|
||||
const uint8_t *row = src + (size_t)y * row_b;
|
||||
pack_line(gbuf + 3, line_payload, head_dots, row, wd, left_dots,
|
||||
print_dots);
|
||||
memcpy(gbuf, g_hdr, sizeof(g_hdr));
|
||||
ptouch_fill_gf_header(gbuf, line_payload);
|
||||
v = ptouch_bulk_send_job(ctx, gbuf, gf_packet, "raster line");
|
||||
if (v != LIBPTOUCH_OK) {
|
||||
free(gbuf);
|
||||
|
||||
41
src/lib/libptouch_protocol.c
Normal file
41
src/lib/libptouch_protocol.c
Normal file
@@ -0,0 +1,41 @@
|
||||
#include "libptouch_protocol.h"
|
||||
|
||||
size_t ptouch_line_payload_bytes(unsigned head_dots)
|
||||
{
|
||||
return (size_t)((head_dots + 7u) / 8u);
|
||||
}
|
||||
|
||||
void ptouch_fill_gf_header(uint8_t out[3], size_t line_payload_bytes)
|
||||
{
|
||||
out[0] = 0x47u;
|
||||
out[1] = (uint8_t)(line_payload_bytes & 0xFFu);
|
||||
out[2] = (uint8_t)((line_payload_bytes >> 8) & 0xFFu);
|
||||
}
|
||||
|
||||
void ptouch_fill_esc_iz(uint8_t out[13], uint8_t media_kind, uint8_t media_width,
|
||||
uint32_t raster_lines)
|
||||
{
|
||||
uint8_t n2_paper = 0x09u;
|
||||
if (media_kind == 0x03u)
|
||||
n2_paper = 0x00u;
|
||||
else if (media_kind == 0x11u)
|
||||
n2_paper = 0x11u;
|
||||
else if (media_kind == 0x17u)
|
||||
n2_paper = 0x17u;
|
||||
else if (media_kind == 0x13u)
|
||||
n2_paper = 0x13u;
|
||||
|
||||
out[0] = 0x1Bu;
|
||||
out[1] = 0x69u;
|
||||
out[2] = 0x7Au;
|
||||
out[3] = 0x0Eu;
|
||||
out[4] = n2_paper;
|
||||
out[5] = media_width;
|
||||
out[6] = 0x00u;
|
||||
out[7] = (uint8_t)(raster_lines & 0xFFu);
|
||||
out[8] = (uint8_t)((raster_lines >> 8) & 0xFFu);
|
||||
out[9] = (uint8_t)((raster_lines >> 16) & 0xFFu);
|
||||
out[10] = (uint8_t)((raster_lines >> 24) & 0xFFu);
|
||||
out[11] = 0x00u; /* first page */
|
||||
out[12] = 0x00u; /* fixed */
|
||||
}
|
||||
12
src/lib/libptouch_protocol.h
Normal file
12
src/lib/libptouch_protocol.h
Normal file
@@ -0,0 +1,12 @@
|
||||
#ifndef LIBPTOUCH_PROTOCOL_H
|
||||
#define LIBPTOUCH_PROTOCOL_H
|
||||
|
||||
#include <stddef.h>
|
||||
#include <stdint.h>
|
||||
|
||||
size_t ptouch_line_payload_bytes(unsigned head_dots);
|
||||
void ptouch_fill_gf_header(uint8_t out[3], size_t line_payload_bytes);
|
||||
void ptouch_fill_esc_iz(uint8_t out[13], uint8_t media_kind, uint8_t media_width,
|
||||
uint32_t raster_lines);
|
||||
|
||||
#endif /* LIBPTOUCH_PROTOCOL_H */
|
||||
@@ -9,6 +9,7 @@
|
||||
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
|
||||
libptouch_err_t libptouch_get_status(libptouch_ctx *ctx, uint8_t *status)
|
||||
{
|
||||
@@ -28,18 +29,26 @@ libptouch_err_t libptouch_get_status(libptouch_ctx *ctx, uint8_t *status)
|
||||
static const uint8_t req[] = { 0x1B, 0x69, 0x53 };
|
||||
|
||||
int r = LIBUSB_ERROR_OTHER;
|
||||
for (int attempt = 0; attempt < 2; attempt++) {
|
||||
for (int attempt = 0; attempt < 5; attempt++) {
|
||||
/* First status read after reconnect can fail on some P-touch units. */
|
||||
(void)libusb_clear_halt(h, ctx->bulk_in_ep);
|
||||
(void)libusb_clear_halt(h, ctx->bulk_out_ep);
|
||||
if (attempt > 0) {
|
||||
(void)libusb_clear_halt(h, ctx->bulk_in_ep);
|
||||
(void)libusb_clear_halt(h, ctx->bulk_out_ep);
|
||||
usleep(120000); /* 120ms backoff */
|
||||
}
|
||||
r = ptouch_bulk_out(h, ctx->bulk_out_ep, init, 2, 5000u);
|
||||
if (r != 0) {
|
||||
if (r == LIBUSB_ERROR_IO || r == LIBUSB_ERROR_PIPE ||
|
||||
r == LIBUSB_ERROR_TIMEOUT)
|
||||
continue;
|
||||
ptouch_set_error_usb(ctx, r, "bulk OUT ESC @");
|
||||
return LIBPTOUCH_ERR_USB;
|
||||
}
|
||||
r = ptouch_bulk_out(h, ctx->bulk_out_ep, req, 3, 5000u);
|
||||
if (r != 0) {
|
||||
if (r == LIBUSB_ERROR_IO || r == LIBUSB_ERROR_PIPE ||
|
||||
r == LIBUSB_ERROR_TIMEOUT)
|
||||
continue;
|
||||
ptouch_set_error_usb(ctx, r, "bulk OUT ESC i S");
|
||||
return LIBPTOUCH_ERR_USB;
|
||||
}
|
||||
@@ -47,7 +56,8 @@ libptouch_err_t libptouch_get_status(libptouch_ctx *ctx, uint8_t *status)
|
||||
(int)LIBPTOUCH_STATUS_LENGTH, 5000u);
|
||||
if (r == 0)
|
||||
break;
|
||||
if (r != LIBUSB_ERROR_IO && r != LIBUSB_ERROR_PIPE)
|
||||
if (r != LIBUSB_ERROR_IO && r != LIBUSB_ERROR_PIPE &&
|
||||
r != LIBUSB_ERROR_TIMEOUT)
|
||||
break;
|
||||
}
|
||||
if (r != 0) {
|
||||
|
||||
Reference in New Issue
Block a user