diff --git a/0001-libcupsfilters-In-pdftopdf-fix-cropping-with-long-ed.patch b/0001-libcupsfilters-In-pdftopdf-fix-cropping-with-long-ed.patch new file mode 100644 index 0000000..190e198 --- /dev/null +++ b/0001-libcupsfilters-In-pdftopdf-fix-cropping-with-long-ed.patch @@ -0,0 +1,36 @@ +From d4ae67231047ac6a4d532d96ddf2234465d57c1d Mon Sep 17 00:00:00 2001 +From: Till Kamppeter +Date: Sun, 27 Mar 2022 17:51:12 +0200 +Subject: [PATCH] libcupsfilters: In pdftopdf() fix cropping with + long-edge-first + +If the printer takes the paper long-edge-first (lasers and inkjets +usually take it short-edge first, but roll-fed large-formats or label +printers also take long-edge-first) the cropping of the page image for +crop-to-fit (print-scaling=none) and fill (print-scaling=fill) by the +pdftopdf() filter function did not work correctly. This is fixed now. + +Issue #454 + +(manually backported from commit 70e0f47082550ae212f6b2741a447cbae17a857d) +--- + filter/pdftopdf/qpdf_pdftopdf_processor.cc | 3 ++- + 1 file changed, 2 insertions(+), 1 deletion(-) + +diff --git a/filter/pdftopdf/qpdf_pdftopdf_processor.cc b/filter/pdftopdf/qpdf_pdftopdf_processor.cc +index 621c76880..a330e825f 100644 +--- a/filter/pdftopdf/qpdf_pdftopdf_processor.cc ++++ b/filter/pdftopdf/qpdf_pdftopdf_processor.cc +@@ -193,7 +193,8 @@ Rotation QPDF_PDFTOPDF_PageHandle::crop(const PageRect &cropRect,Rotation orient + double final_w,final_h; //Width and height of cropped image. + + Rotation pageRot = getRotate(page); +- if(pageRot==ROT_0||pageRot==ROT_180) ++ if (((pageRot == ROT_0 || pageRot == ROT_180) && pageWidth <= pageHeight) || ++ ((pageRot == ROT_90 || pageRot == ROT_270) && pageWidth > pageHeight)) + { + std::swap(pageHeight,pageWidth); + } +-- +2.47.1 + diff --git a/0001-pdftopdf-Add-2-tolerance-for-input-size-larger-than-.patch b/0001-pdftopdf-Add-2-tolerance-for-input-size-larger-than-.patch new file mode 100644 index 0000000..b8035f1 --- /dev/null +++ b/0001-pdftopdf-Add-2-tolerance-for-input-size-larger-than-.patch @@ -0,0 +1,39 @@ +From ddd9da98fd86be3730747f1da921b22f2452062c Mon Sep 17 00:00:00 2001 +From: Till Kamppeter +Date: Sat, 5 Feb 2022 16:22:10 -0300 +Subject: [PATCH] pdftopdf: Add 2% tolerance for input size larger than output + page + +When "print-scaling=auto" or "print-scaling=auto-fit" is used, the +input pages are scaled when they do not fit into the output page +size. Often input ad out page sizes are supposed to be equal, for +example both A4, but rounding errors could make the input considered +larger and unnecessarily scaled. + +Therefore we add 2% of tolerance before considering an input page too +large. + +(manually backported commit e541dc698f38aaa522f99e23b57c695da2981c29) +--- + NEWS | 6 ++++++ + filter/pdftopdf/pdftopdf_processor.cc | 4 ++-- + 2 files changed, 8 insertions(+), 2 deletions(-) + +diff --git a/filter/pdftopdf/pdftopdf_processor.cc b/filter/pdftopdf/pdftopdf_processor.cc +index e297a246e..28643163f 100644 +--- a/filter/pdftopdf/pdftopdf_processor.cc ++++ b/filter/pdftopdf/pdftopdf_processor.cc +@@ -195,8 +195,8 @@ bool processPDFTOPDF(PDFTOPDF_Processor &proc,ProcessingParameters ¶m) // {{ + for (int i = 0; i < (int)pages.size(); i ++) + { + PageRect r = pages[i]->getRect(); +- int w = r.width; +- int h = r.height; ++ int w = r.width * 100 / 102; // 2% of tolerance ++ int h = r.height * 100 / 102; + if ((w > param.page.width || h > param.page.height) && + (h > param.page.width || w > param.page.height)) + { +-- +2.47.1 + diff --git a/0001-pdftopdf-Fixed-print-scaling-and-N-up-for-asymmetric.patch b/0001-pdftopdf-Fixed-print-scaling-and-N-up-for-asymmetric.patch new file mode 100644 index 0000000..3de2594 --- /dev/null +++ b/0001-pdftopdf-Fixed-print-scaling-and-N-up-for-asymmetric.patch @@ -0,0 +1,480 @@ +From 207314fc3dbcfde829f5d0082c6a23ad695d34b9 Mon Sep 17 00:00:00 2001 +From: Till Kamppeter +Date: Tue, 1 Feb 2022 23:00:46 -0300 +Subject: [PATCH] pdftopdf: Fixed print-scaling and N-up for asymmetric margins + and files with differently-sized pages + +(manually backported commit 4aaf23aae3695348532e295859f001704a33ebad + and commit 31dfcae961ca737b7166cd6a3e7d4a30cd19f9e8) +--- + NEWS | 8 + + filter/pdftopdf/pdftopdf_processor.cc | 362 ++++++++++----------- + filter/pdftopdf/qpdf_pdftopdf_processor.cc | 9 +- + 3 files changed, 189 insertions(+), 190 deletions(-) + +diff --git a/filter/pdftopdf/pdftopdf_processor.cc b/filter/pdftopdf/pdftopdf_processor.cc +index 6d2d32ff5..e297a246e 100644 +--- a/filter/pdftopdf/pdftopdf_processor.cc ++++ b/filter/pdftopdf/pdftopdf_processor.cc +@@ -175,31 +175,46 @@ bool processPDFTOPDF(PDFTOPDF_Processor &proc,ProcessingParameters ¶m) // {{ + } + const int numPages=std::max(shuffle.size(),pages.size()); + ++ fprintf(stderr, "DEBUG: pdftopdf: \"print-scaling\" IPP attribute: %s\n", ++ (param.autoprint ? "auto" : ++ (param.autofit ? "auto-fit" : ++ (param.fitplot ? "fit" : ++ (param.fillprint ? "fill" : ++ (param.cropfit ? "none" : ++ "Not defined, should never happen")))))); ++ + if(param.autoprint||param.autofit){ + bool margin_defined = true; + bool document_large = false; + int pw = param.page.right-param.page.left; + int ph = param.page.top-param.page.bottom; +- int w=0,h=0; +- Rotation tempRot=param.orientation; +- PageRect r= pages[0]->getRect(); +- w = r.width; +- h = r.height; + +- if(tempRot==ROT_90||tempRot==ROT_270) ++ if ((param.page.width == pw) && (param.page.height == ph)) ++ margin_defined = false; ++ ++ for (int i = 0; i < (int)pages.size(); i ++) + { +- std::swap(w,h); ++ PageRect r = pages[i]->getRect(); ++ int w = r.width; ++ int h = r.height; ++ if ((w > param.page.width || h > param.page.height) && ++ (h > param.page.width || w > param.page.height)) ++ { ++ fprintf(stderr, ++ "DEBUG: pdftopdf: Page %d too large for output page size, scaling pages to fit.\n", ++ i + 1); ++ document_large = true; ++ } + } +- if(w>=pw||h>=ph) ++ if (param.fidelity) ++ fprintf(stderr, ++ "DEBUG: pdftopdf: \"ipp-attribute-fidelity\" IPP attribute is set, scaling pages to fit.\n"); ++ ++ if (param.autoprint) + { +- document_large = true; +- } +- if((param.page.width==pw)&& +- (param.page.height==ph)) +- margin_defined = false; +- if(param.autoprint){ +- if(param.fidelity||document_large) { +- if(margin_defined) ++ if (param.fidelity || document_large) ++ { ++ if (margin_defined) + param.fitplot = true; + else + param.fillprint = true; +@@ -215,204 +230,177 @@ bool processPDFTOPDF(PDFTOPDF_Processor &proc,ProcessingParameters ¶m) // {{ + } + } + ++ fprintf(stderr, "DEBUG: pdftopdf: Print scaling mode: %s\n", ++ (param.fitplot ? ++ "Scale to fit printable area" : ++ (param.fillprint ? ++ "Scale to fill page and crop" : ++ (param.cropfit ? ++ "Do not scale, center, crop if needed" : ++ "Not defined, should never happen")))); ++ ++ // In Crop mode we do not scale the original document, it should keep the ++ // exact same size. With N-Up it should be scaled to fit exacly the halves, ++ // quarters, ... of the sheet, regardless of unprintable margins. ++ // Therefore we remove the unprintable margins to do all the math without ++ // them. ++ if (param.cropfit) ++ { ++ param.page.left = 0; ++ param.page.bottom = 0; ++ param.page.right = param.page.width; ++ param.page.top = param.page.height; ++ } ++ + if(param.fillprint||param.cropfit){ +- fprintf(stderr,"[DEBUG]: Cropping input pdf and Enabling fitplot.\n"); +- if(param.noOrientation&&pages.size()) +- { +- bool land = pages[0]->is_landscape(param.orientation); +- if(land) +- param.orientation = param.normal_landscape; +- } + for(int i=0;i<(int)pages.size();i++) + { + std::shared_ptr page = pages[i]; +- page->crop(param.page,param.orientation,param.xpos,param.ypos,!param.cropfit); ++ Rotation orientation = param.orientation; ++ if (param.noOrientation && ++ page->is_landscape(param.orientation)) ++ orientation = param.normal_landscape; ++ page->crop(param.page, orientation, param.xpos, param.ypos, ++ !param.cropfit); + } +- param.fitplot = 1; ++ if (param.fillprint) ++ param.fitplot = true; + } + + std::shared_ptr curpage; + int outputpage=0; + int outputno=0; + +- if ((param.nup.nupX==1)&&(param.nup.nupY==1)&&(!param.fitplot)) { +- // TODO? fitplot also without xobject? +- /* +- param.nup.width=param.page.width; +- param.nup.height=param.page.height; +- */ ++ if ((param.nup.nupX == 1) && (param.nup.nupY == 1) && !param.fitplot) ++ { ++ param.nup.width = param.page.width; ++ param.nup.height = param.page.height; ++ } ++ else ++ { ++ param.nup.width = param.page.right - param.page.left; ++ param.nup.height = param.page.top - param.page.bottom; ++ } + +- for (int iA=0;iArotate(param.normal_landscape); ++ param.orientation = param.orientation + param.normal_landscape; ++ // TODO? better ++ if (param.nup.nupX != 1 || param.nup.nupY != 1 || param.fitplot) ++ { ++ xpos = param.page.height - param.page.top; ++ ypos = param.page.left; ++ } ++ std::swap(param.page.width, param.page.height); ++ std::swap(param.nup.width, param.nup.height); ++ } ++ else ++ { ++ if (param.nup.nupX != 1 || param.nup.nupY != 1 || param.fitplot) ++ { ++ xpos = param.page.left; ++ ypos = param.page.bottom; // for whole page... TODO from position... ++ } ++ } + +- if (shuffle[iA]>=numOrigPages) { +- // add empty page as filler +- proc.add_page(proc.new_page(param.page.width,param.page.height),param.reverse); ++ NupState nupstate(param.nup); ++ NupPageEdit pgedit; ++ for (int iA=0;iA page; ++ if (shuffle[iA] >= numOrigPages) ++ // add empty page as filler ++ page=proc.new_page(param.page.width,param.page.height); ++ else ++ page=pages[shuffle[iA]]; ++ ++ PageRect rect; ++ rect = page->getRect(); ++ //rect.dump(); ++ ++ bool newPage=nupstate.nextPage(rect.width,rect.height,pgedit); ++ if (newPage) { ++ if ((curpage)&&(param.withPage(outputpage))) { ++ curpage->rotate(param.orientation); ++ if (param.mirror) ++ curpage->mirror(); ++ // TODO? update rect? --- not needed any more ++ proc.add_page(curpage,param.reverse); // reverse -> insert at beginning ++ // Log page in /var/log/cups/page_log + outputno++; +- continue; // no border, etc. +- } +- auto page=pages[shuffle[iA]]; +- +- page->rotate(param.orientation); +- +- if (param.mirror) { +- page->mirror(); +- } +- +- if (!param.pageLabel.empty()) { +- page->add_label(param.page, param.pageLabel); ++ if (param.page_logging == 1) ++ fprintf(stderr, "PAGE: %d %d\n", outputno, ++ param.copies_to_be_logged); + } +- +- // place border +- if ((param.border!=BorderType::NONE)&&(iAgetRect(); +- +- rect.left+=param.page.left; +- rect.bottom+=param.page.bottom; +- rect.top-=param.page.top; +- rect.right-=param.page.right; +- // width,height not needed for add_border_rect (FIXME?) +- +- page->add_border_rect(rect,param.border,1.0); +-#else // this is what pstops does +- page->add_border_rect(param.page,param.border,1.0); +-#endif +- } +- +- proc.add_page(page,param.reverse); // reverse -> insert at beginning +- outputno++; ++ curpage=proc.new_page(param.page.width,param.page.height); ++ outputpage++; + } +- } else { +- param.nup.width=param.page.right-param.page.left; +- param.nup.height=param.page.top-param.page.bottom; +- +- double xpos=param.page.left, +- ypos=param.page.bottom; // for whole page... TODO from position... +- +- const bool origls=param.nup.landscape; +- if ((param.orientation==ROT_90)||(param.orientation==ROT_270)) { +- std::swap(param.nup.nupX,param.nup.nupY); +- param.nup.landscape=!param.nup.landscape; +- param.orientation=param.orientation-param.normal_landscape; +- } +- if (param.nup.landscape) { +- // pages[iA]->rotate(param.normal_landscape); +- param.orientation=param.orientation+param.normal_landscape; +- // TODO? better +- xpos=param.page.bottom; +- ypos=param.page.width - param.page.right; +- std::swap(param.page.width,param.page.height); +- std::swap(param.nup.width,param.nup.height); ++ if (shuffle[iA]>=numOrigPages) { ++ continue; + } + +- NupState nupstate(param.nup); +- NupPageEdit pgedit; +- for (int iA=0;iA page; +- if (shuffle[iA]>=numOrigPages) { +- // add empty page as filler +- page=proc.new_page(param.page.width,param.page.height); +- } else { +- page=pages[shuffle[iA]]; +- } +- +- PageRect rect; +- if (param.fitplot) { +- rect=page->getRect(); +- } else { +- rect.width=param.page.width; +- rect.height=param.page.height; +- +- // TODO? better +- if (origls) { +- std::swap(rect.width,rect.height); +- } +- +- rect.left=0; +- rect.bottom=0; +- rect.right=rect.width; +- rect.top=rect.height; +- } +- // rect.dump(); +- +- bool newPage=nupstate.nextPage(rect.width,rect.height,pgedit); +- if (newPage) { +- if ((curpage)&&(param.withPage(outputpage))) { +- curpage->rotate(param.orientation); +- if (param.mirror) { +- curpage->mirror(); +- // TODO? update rect? --- not needed any more +- } +- proc.add_page(curpage,param.reverse); // reverse -> insert at beginning +- // Log page in /var/log/cups/page_log +- outputno++; +- if (param.page_logging == 1) +- fprintf(stderr, "PAGE: %d %d\n", outputno, +- param.copies_to_be_logged); +- } +- curpage=proc.new_page(param.page.width,param.page.height); +- outputpage++; +- } +- if (shuffle[iA]>=numOrigPages) { +- continue; +- } +- +- if (param.border!=BorderType::NONE) { +- // TODO FIXME... border gets cutted away, if orignal page had wrong size +- // page->"uncrop"(rect); // page->setMedia() +- // Note: currently "fixed" in add_subpage(...&rect); +- page->add_border_rect(rect,param.border,1.0/pgedit.scale); +- } ++ if (param.border!=BorderType::NONE) { ++ // TODO FIXME... border gets cutted away, if orignal page had wrong size ++ // page->"uncrop"(rect); // page->setMedia() ++ // Note: currently "fixed" in add_subpage(...&rect); ++ page->add_border_rect(rect,param.border,1.0/pgedit.scale); ++ } + +- if (!param.pageLabel.empty()) { +- page->add_label(param.page, param.pageLabel); +- } ++ if (!param.pageLabel.empty()) { ++ page->add_label(param.page, param.pageLabel); ++ } + +- if (!param.fitplot) { +- curpage->add_subpage(page,pgedit.xpos+xpos,pgedit.ypos+ypos,pgedit.scale,&rect); +- } else { +- if(param.cropfit){ +- double xpos2 = (param.page.right-param.page.left-(page->getRect().width))/2; +- double ypos2 = (param.page.top-param.page.bottom-(page->getRect().height))/2; +- if(param.orientation==ROT_270||param.orientation==ROT_90) +- { +- xpos2 = (param.page.right-param.page.left-(page->getRect().height))/2; +- ypos2 = (param.page.top-param.page.bottom-(page->getRect().width))/2; +- curpage->add_subpage(page,ypos2+param.page.bottom,xpos2+param.page.left,1); +- }else{ +- curpage->add_subpage(page,xpos2+param.page.left,ypos2+param.page.bottom,1); +- } +- } +- else +- curpage->add_subpage(page,pgedit.xpos+xpos,pgedit.ypos+ypos,pgedit.scale); ++ if (param.cropfit) ++ { ++ if ((param.nup.nupX == 1) && (param.nup.nupY == 1)) ++ { ++ double xpos2, ypos2; ++ if (param.orientation == ROT_270 || param.orientation == ROT_90) ++ { ++ xpos2 = (param.page.width - (page->getRect().height)) / 2; ++ ypos2 = (param.page.height - (page->getRect().width)) / 2; ++ curpage->add_subpage(page, ypos2 + xpos, xpos2 + ypos, 1); ++ } ++ else ++ { ++ xpos2 = (param.page.width - (page->getRect().width)) / 2; ++ ypos2 = (param.page.height - (page->getRect().height)) / 2; ++ curpage->add_subpage(page, xpos2 + xpos, ypos2 + ypos, 1); ++ } + } ++ else ++ curpage->add_subpage(page,pgedit.xpos+xpos,pgedit.ypos+ypos,pgedit.scale); ++ } ++ else ++ curpage->add_subpage(page, pgedit.xpos + xpos, pgedit.ypos + ypos, ++ pgedit.scale); + + #ifdef DEBUG +- if (auto dbg=dynamic_cast(curpage.get())) { +- // dbg->debug(pgedit.sub,xpos,ypos); +- } ++ if (auto dbg=dynamic_cast(curpage.get())) { ++ dbg->debug(pgedit.sub,xpos,ypos); ++ } + #endif + +- // pgedit.dump(); +- } +- if ((curpage)&&(param.withPage(outputpage))) { +- curpage->rotate(param.orientation); +- if (param.mirror) { +- curpage->mirror(); +- } +- proc.add_page(curpage,param.reverse); // reverse -> insert at beginning +- // Log page in /var/log/cups/page_log +- outputno ++; +- if (param.page_logging == 1) +- fprintf(stderr, "PAGE: %d %d\n", outputno, param.copies_to_be_logged); ++ // pgedit.dump(); ++ } ++ if ((curpage)&&(param.withPage(outputpage))) { ++ curpage->rotate(param.orientation); ++ if (param.mirror) { ++ curpage->mirror(); + } ++ proc.add_page(curpage,param.reverse); // reverse -> insert at beginning ++ // Log page in /var/log/cups/page_log ++ outputno ++; ++ if (param.page_logging == 1) ++ fprintf(stderr, "PAGE: %d %d\n", outputno, param.copies_to_be_logged); + } + + if ((param.evenDuplex || !param.oddPages) && (outputno & 1)) { +diff --git a/filter/pdftopdf/qpdf_pdftopdf_processor.cc b/filter/pdftopdf/qpdf_pdftopdf_processor.cc +index d06ca5f77..621c76880 100644 +--- a/filter/pdftopdf/qpdf_pdftopdf_processor.cc ++++ b/filter/pdftopdf/qpdf_pdftopdf_processor.cc +@@ -179,6 +179,7 @@ void QPDF_PDFTOPDF_PageHandle::add_border_rect(const PageRect &_rect,BorderType + Rotation QPDF_PDFTOPDF_PageHandle::crop(const PageRect &cropRect,Rotation orientation,Position xpos,Position ypos,bool scale) + { + page.assertInitialized(); ++ Rotation save_rotate = getRotate(page); + if(orientation==ROT_0||orientation==ROT_180) + page.replaceOrRemoveKey("/Rotate",makeRotate(ROT_90)); + else +@@ -209,8 +210,8 @@ Rotation QPDF_PDFTOPDF_PageHandle::crop(const PageRect &cropRect,Rotation orient + } + } + else{ +- final_w = std::min(width,pageWidth); +- final_h = std::min(height,pageHeight); ++ final_w = pageWidth; ++ final_h = pageHeight; + } + fprintf(stderr,"After Cropping: %lf %lf %lf %lf\n",width,height,final_w,final_h); + double posw = (width-final_w)/2, +@@ -235,13 +236,14 @@ Rotation QPDF_PDFTOPDF_PageHandle::crop(const PageRect &cropRect,Rotation orient + //Cropping. + // TODO: Borders are covered by the image. buffer space? + page.replaceKey("/TrimBox",makeBox(currpage.left,currpage.bottom,currpage.right,currpage.top)); +- page.replaceOrRemoveKey("/Rotate",makeRotate(ROT_0)); ++ page.replaceOrRemoveKey("/Rotate",makeRotate(save_rotate)); + return getRotate(page); + } + + bool QPDF_PDFTOPDF_PageHandle::is_landscape(Rotation orientation) + { + page.assertInitialized(); ++ Rotation save_rotate = getRotate(page); + if(orientation==ROT_0||orientation==ROT_180) + page.replaceOrRemoveKey("/Rotate",makeRotate(ROT_90)); + else +@@ -250,6 +252,7 @@ bool QPDF_PDFTOPDF_PageHandle::is_landscape(Rotation orientation) + PageRect currpage= getBoxAsRect(getTrimBox(page)); + double width = currpage.right-currpage.left; + double height = currpage.top-currpage.bottom; ++ page.replaceOrRemoveKey("/Rotate",makeRotate(save_rotate)); + if(width>height) + return true; + return false; +-- +2.47.1 + diff --git a/0001-pdftopdf-Set-a-default-for-print-scaling.patch b/0001-pdftopdf-Set-a-default-for-print-scaling.patch new file mode 100644 index 0000000..456110f --- /dev/null +++ b/0001-pdftopdf-Set-a-default-for-print-scaling.patch @@ -0,0 +1,36 @@ +From 02e95494b49775e9e314d32d78568058d67c9b25 Mon Sep 17 00:00:00 2001 +From: Till Kamppeter +Date: Thu, 3 Feb 2022 19:03:00 -0300 +Subject: [PATCH] pdftopdf: Set a default for "print-scaling" + +If no setting for "print-scaling" is provided via IPP attribute or +command line option there is no default value for it and we get +undefined behavior and also a scary "should never happen" message in +the log. + +Now we check after parsing the appropriate IPP attributes and command +line options whether a setting was provided. If not, we use "auto". + +(manually backported commit dce6292f81b6872656cfcc46dac2ec101c32bf4a) +--- + NEWS | 2 ++ + filter/pdftopdf/pdftopdf.cc | 3 +++ + 2 files changed, 5 insertions(+) + +diff --git a/filter/pdftopdf/pdftopdf.cc b/filter/pdftopdf/pdftopdf.cc +index c6f9d5004..7a684a260 100644 +--- a/filter/pdftopdf/pdftopdf.cc ++++ b/filter/pdftopdf/pdftopdf.cc +@@ -357,6 +357,9 @@ void getParameters(ppd_file_t *ppd,int num_options,cups_option_t *options,Proces + param.cropfit=1; + } + } ++ if (!param.autoprint && !param.autofit && !param.fitplot && ++ !param.fillprint && !param.cropfit) ++ param.autoprint = true; + } + + if (ppd && (ppd->landscape < 0)) { // direction the printer rotates landscape (90 or -90) +-- +2.47.1 + diff --git a/cups-filters.spec b/cups-filters.spec index bbfb486..034180d 100644 --- a/cups-filters.spec +++ b/cups-filters.spec @@ -11,7 +11,7 @@ Summary: OpenPrinting CUPS filters and backends Name: cups-filters Version: 1.28.7 -Release: 19%{?dist} +Release: 20%{?dist} # For a breakdown of the licensing, see COPYING file # GPLv2: filters: commandto*, imagetoraster, pdftops, rasterto*, @@ -50,6 +50,15 @@ Patch07: cups-filters-CVE-2024-47175.patch Patch08: 0001-cfGetPrinterAttributes5-Validate-response-attributes.patch # RHEL-17124 [cups-browsed] Prints to remote RAW queues are converted to PDF documents Patch09: 0001-Do-not-generate-PPD-for-remote-raw-queues-44.patch +# RHEL-70015 Double-sided printing issues on Thunderbird or Firefox // HP printers +# Patches: 0001-pdftopdf-Fixed-print-scaling-and-N-up-for-asymmetric.patcj +# 0001-pdftopdf-Set-a-default-for-print-scaling.patch +# 0001-pdftopdf-Add-2-tolerance-for-input-size-larger-than-.patch +# 0001-libcupsfilters-In-pdftopdf-fix-cropping-with-long-ed.patch +Patch10: 0001-pdftopdf-Fixed-print-scaling-and-N-up-for-asymmetric.patch +Patch11: 0001-pdftopdf-Set-a-default-for-print-scaling.patch +Patch12: 0001-pdftopdf-Add-2-tolerance-for-input-size-larger-than-.patch +Patch13: 0001-libcupsfilters-In-pdftopdf-fix-cropping-with-long-ed.patch # autogen.sh @@ -456,6 +465,9 @@ fi %endif %changelog +* Thu Dec 12 2024 Zdenek Dohnal - 1.28.7-20 +- RHEL-70015 Double-sided printing issues on Thunderbird or Firefox // HP printers + * Tue Oct 15 2024 Zdenek Dohnal - 1.28.7-19 - RHEL-17124 [cups-browsed] Prints to remote RAW queues are converted to PDF documents