From 931c677443cccd846fb2856c630a4dd81b7f0079 Mon Sep 17 00:00:00 2001 From: Peter De Wachter Date: Mon, 23 Oct 2017 22:09:29 +0200 Subject: [PATCH] Get rid of iconv() PJL has a job name field. I don't think this is used for anything, but just in case I want to put something human-readable there. But it's also probably a good idea to keep this pure ASCII text. Previously I used iconv() to convert the CUPS job name to ASCII. But: - on FreeBSD, an explicit "-liconv" seems to be necessary (#10), - there's an autoconf macro to deal with this, but it's tangled up with gettext and it seems like a hassle, - the //TRANSLIT//IGNORE feature I used seems to be a GNU extension and probably won't work on other systems anyway. So instead, let's just concatenate some job information together, dropping any non-ASCII component. That's probably good enough. --- src/main.cc | 67 +++++++++++++++++++++++++---------------------------- 1 file changed, 31 insertions(+), 36 deletions(-) diff --git a/src/main.cc b/src/main.cc index db238eb..a185dec 100644 --- a/src/main.cc +++ b/src/main.cc @@ -18,7 +18,6 @@ #include #include #include -#include #include #include #include @@ -51,41 +50,37 @@ bool next_line(uint8_t *buf) { return cupsRasterReadPixels(ras, buf, bytes) == bytes; } -// POSIX says the second argument of iconv has type 'char **', but -// some systems have 'const char **'. This class is used to work -// around this incompatibility. -class sloppy_ptr { - public: - explicit sloppy_ptr(const char **ptr): ptr_(ptr) { } - operator const char **() { return ptr_; } - operator char **() { return const_cast(ptr_); } - private: - const char **ptr_; -}; +bool plain_ascii_string(const char *str) { + bool result = true; + for (; result && *str; str++) { + result = *str >= 32 && *str <= 126; + } + return result; +} -std::string ascii_job_name(const char *job_name, const char *charset) { - if (job_name && charset) { - iconv_t cd = iconv_open("ASCII//TRANSLIT//IGNORE", charset); - if (cd != (iconv_t) -1) { - char ascii[80]; - const char *in_ptr = job_name; - size_t in_left = strlen(job_name); - char *out_ptr = ascii; - size_t out_left = sizeof(ascii) - 1; - while (in_left > 0) { - size_t err = iconv(cd, - sloppy_ptr(&in_ptr), &in_left, - &out_ptr, &out_left); - if (err == (size_t) -1) { - break; - } +std::string ascii_job_name(const char *job_id, const char *job_user, const char *job_name) { + std::array parts = {{ + job_id, + job_user, + job_name + }}; + std::string result; + for (const char *part : parts) { + if (*part && plain_ascii_string(part)) { + if (!result.empty()) { + result += '/'; } - *out_ptr = 0; - iconv_close(cd); - return ascii; + result += part; } } - return "CUPS"; + if (result.empty()) { + result = "brlaser"; + } + const int max_size = 79; + if (result.size() > max_size) { + result.resize(max_size); + } + return result; } page_params build_page_params() { @@ -139,13 +134,13 @@ int main(int argc, char *argv[]) { fprintf(stderr, "INFO: This program is a CUPS filter. It is not intended to be run manually.\n"); return 1; } - // const char *job_id = argv[1]; - // const char *job_user = argv[2]; + const char *job_id = argv[1]; + const char *job_user = argv[2]; const char *job_name = argv[3]; // const int job_copies = atoi(argv[4]); // const char *job_options = argv[5]; const char *job_filename = argv[6]; - const char *job_charset = getenv("CHARSET"); + // const char *job_charset = getenv("CHARSET"); setlocale(LC_ALL, ""); @@ -169,7 +164,7 @@ int main(int argc, char *argv[]) { int pages = 0; { - job job(stdout, ascii_job_name(job_name, job_charset)); + job job(stdout, ascii_job_name(job_id, job_user, job_name)); while (!interrupted && cupsRasterReadHeader2(ras, &header)) { if (header.cupsBitsPerPixel != 1 || header.cupsBitsPerColor != 1