3dc527
From 64aa150a92ccb082db6a3383fa734a6ac91cf1bf Mon Sep 17 00:00:00 2001
3dc527
From: Marek Kasik <mkasik@redhat.com>
3dc527
Date: Tue, 30 Apr 2019 18:47:44 +0200
3dc527
Subject: [PATCH] PSOutputDev: Don't read outside of image buffer
3dc527
3dc527
Check whether input image is RGB or BGR to not treat
3dc527
it as CMYK in those cases in PSOutputDev::checkPageSlice().
3dc527
3dc527
Fixes #751
3dc527
---
3dc527
 poppler/PSOutputDev.cc | 248 ++++++++++++++++++++++++++++++++---------
3dc527
 1 file changed, 196 insertions(+), 52 deletions(-)
3dc527
3dc527
diff --git a/poppler/PSOutputDev.cc b/poppler/PSOutputDev.cc
3dc527
index 0d201835..155a8cbe 100644
3dc527
--- a/poppler/PSOutputDev.cc
3dc527
+++ b/poppler/PSOutputDev.cc
3dc527
@@ -3342,13 +3342,21 @@ GBool PSOutputDev::checkPageSlice(Page *
3dc527
       }
3dc527
       break;
3dc527
     case psLevel1Sep:
3dc527
+      GfxColor inputColor;
3dc527
+      GfxCMYK cmyk;
3dc527
+      unsigned char cmykColor[4];
3dc527
+      GfxDeviceRGBColorSpace *rgbCS;
3dc527
+      SplashColorMode colorMode;
3dc527
+
3dc527
+      colorMode = bitmap->getMode();
3dc527
+
3dc527
       useBinary = globalParams->getPSBinary();
3dc527
       p = bitmap->getDataPtr();
3dc527
       // Check for an all gray image
3dc527
       isGray = gTrue;
3dc527
       for (y = 0; y < h; ++y) {
3dc527
 	for (x = 0; x < w; ++x) {
3dc527
-	  if (p[4*x] != p[4*x + 1] || p[4*x] != p[4*x + 2]) {
3dc527
+	  if (p[numComps*x] != p[numComps*x + 1] || p[numComps*x] != p[numComps*x + 2]) {
3dc527
 	    isGray = gFalse;
3dc527
 	    y = h;
3dc527
 	    break;
3dc527
@@ -3365,7 +3373,9 @@ GBool PSOutputDev::checkPageSlice(Page *
3dc527
       col[0] = col[1] = col[2] = col[3] = 0;
3dc527
       if (isGray) {
3dc527
         int g;
3dc527
-        if ((psProcessBlack & processColors) == 0) {
3dc527
+        if ((psProcessBlack & processColors) == 0 &&
3dc527
+            colorMode != splashModeRGB8 &&
3dc527
+            colorMode != splashModeBGR8) {
3dc527
 	  // Check if the image uses black
3dc527
 	  for (y = 0; y < h; ++y) {
3dc527
 	    for (x = 0; x < w; ++x) {
3dc527
@@ -3379,59 +3389,23 @@ GBool PSOutputDev::checkPageSlice(Page *
3dc527
 	  }
3dc527
           p = bitmap->getDataPtr() + (h - 1) * bitmap->getRowSize();
3dc527
         }
3dc527
-        for (y = 0; y < h; ++y) {
3dc527
-	  if (useBinary) {
3dc527
-	    // Binary gray image
3dc527
-	    for (x = 0; x < w; ++x) {
3dc527
-	      g = p[4*x] + p[4*x + 3];
3dc527
-	      g = 255 - g;
3dc527
-	      if (g < 0) g = 0;
3dc527
-	      hexBuf[i++] = (Guchar) g;
3dc527
-	      if (i >= 64) {
3dc527
-	        writePSBuf(hexBuf, i);
3dc527
-	        i = 0;
3dc527
-	      }
3dc527
-	    }
3dc527
-	  } else {
3dc527
-	    // Hex gray image
3dc527
-	    for (x = 0; x < w; ++x) {
3dc527
-	      g = p[4*x] + p[4*x + 3];
3dc527
-	      g = 255 - g;
3dc527
-	      if (g < 0) g = 0;
3dc527
-	      digit = g / 16;
3dc527
-	      hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
-	      digit = g % 16;
3dc527
-	      hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
-	      if (i >= 64) {
3dc527
-	        hexBuf[i++] = '\n';
3dc527
-	        writePSBuf(hexBuf, i);
3dc527
-	        i = 0;
3dc527
-	      }
3dc527
-	    }
3dc527
-          }
3dc527
-          p -= bitmap->getRowSize();
3dc527
-        }
3dc527
-      } else if (((psProcessCyan | psProcessMagenta | psProcessYellow | psProcessBlack) & ~processColors) != 0) {
3dc527
-	// Color image, need to check color flags for each dot
3dc527
-        for (y = 0; y < h; ++y) {
3dc527
-          for (comp = 0; comp < 4; ++comp) {
3dc527
+        if (colorMode == splashModeRGB8 || colorMode != splashModeBGR8) {
3dc527
+          for (y = 0; y < h; ++y) {
3dc527
 	    if (useBinary) {
3dc527
-	      // Binary color image
3dc527
+	      // Binary gray image
3dc527
 	      for (x = 0; x < w; ++x) {
3dc527
-	        col[comp] |= p[4*x + comp];
3dc527
-	        hexBuf[i++] = p[4*x + comp];
3dc527
+	        hexBuf[i++] = (Guchar) p[3*x];
3dc527
 	        if (i >= 64) {
3dc527
 	          writePSBuf(hexBuf, i);
3dc527
 	          i = 0;
3dc527
 	        }
3dc527
 	      }
3dc527
 	    } else {
3dc527
-	      // Gray color image
3dc527
+	      // Hex gray image
3dc527
 	      for (x = 0; x < w; ++x) {
3dc527
-	        col[comp] |= p[4*x + comp];
3dc527
-	        digit = p[4*x + comp] / 16;
3dc527
+	        digit = p[3*x] / 16;
3dc527
 	        hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
-	        digit = p[4*x + comp] % 16;
3dc527
+	        digit = p[3*x] % 16;
3dc527
 	        hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
 	        if (i >= 64) {
3dc527
 	          hexBuf[i++] = '\n';
3dc527
@@ -3439,29 +3413,31 @@ GBool PSOutputDev::checkPageSlice(Page *
3dc527
 	          i = 0;
3dc527
 	        }
3dc527
 	      }
3dc527
-	    }
3dc527
+            }
3dc527
           }
3dc527
-          p -= bitmap->getRowSize();
3dc527
-        }
3dc527
-      } else {
3dc527
-	// Color image, do not need to check color flags
3dc527
-        for (y = 0; y < h; ++y) {
3dc527
-          for (comp = 0; comp < 4; ++comp) {
3dc527
+        } else {
3dc527
+          for (y = 0; y < h; ++y) {
3dc527
 	    if (useBinary) {
3dc527
-	      // Binary color image
3dc527
+	      // Binary gray image
3dc527
 	      for (x = 0; x < w; ++x) {
3dc527
-	        hexBuf[i++] = p[4*x + comp];
3dc527
+	        g = p[4*x] + p[4*x + 3];
3dc527
+	        g = 255 - g;
3dc527
+	        if (g < 0) g = 0;
3dc527
+	        hexBuf[i++] = (Guchar) g;
3dc527
 	        if (i >= 64) {
3dc527
 	          writePSBuf(hexBuf, i);
3dc527
 	          i = 0;
3dc527
 	        }
3dc527
 	      }
3dc527
 	    } else {
3dc527
-	      // Hex color image
3dc527
+	      // Hex gray image
3dc527
 	      for (x = 0; x < w; ++x) {
3dc527
-	        digit = p[4*x + comp] / 16;
3dc527
+	        g = p[4*x] + p[4*x + 3];
3dc527
+	        g = 255 - g;
3dc527
+	        if (g < 0) g = 0;
3dc527
+	        digit = g / 16;
3dc527
 	        hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
-	        digit = p[4*x + comp] % 16;
3dc527
+	        digit = g % 16;
3dc527
 	        hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
 	        if (i >= 64) {
3dc527
 	          hexBuf[i++] = '\n';
3dc527
@@ -3469,9 +3445,207 @@ GBool PSOutputDev::checkPageSlice(Page *
3dc527
 	          i = 0;
3dc527
 	        }
3dc527
 	      }
3dc527
-	    }
3dc527
+            }
3dc527
           }
3dc527
-          p -= bitmap->getRowSize();
3dc527
+        }
3dc527
+        p -= bitmap->getRowSize();
3dc527
+      } else if (((psProcessCyan | psProcessMagenta | psProcessYellow | psProcessBlack) & ~processColors) != 0) {
3dc527
+	// Color image, need to check color flags for each dot
3dc527
+        switch (colorMode) {
3dc527
+          case splashModeRGB8:
3dc527
+          case splashModeBGR8:
3dc527
+            rgbCS = new GfxDeviceRGBColorSpace();
3dc527
+            for (y = 0; y < h; ++y) {
3dc527
+              for (comp = 0; comp < 4; ++comp) {
3dc527
+	        if (useBinary) {
3dc527
+	          // Binary color image
3dc527
+	          for (x = 0; x < w; ++x) {
3dc527
+	            if (likely(colorMode == splashModeRGB8)) {
3dc527
+                      inputColor.c[0] = byteToCol(p[3*x + 0]);
3dc527
+                      inputColor.c[1] = byteToCol(p[3*x + 1]);
3dc527
+                      inputColor.c[2] = byteToCol(p[3*x + 2]);
3dc527
+                    } else {
3dc527
+                      inputColor.c[0] = byteToCol(p[3*x + 2]);
3dc527
+                      inputColor.c[1] = byteToCol(p[3*x + 1]);
3dc527
+                      inputColor.c[2] = byteToCol(p[3*x + 0]);
3dc527
+                    }
3dc527
+                    rgbCS->getCMYK(&inputColor, &cmyk);
3dc527
+                    cmykColor[0] = colToByte(cmyk.c);
3dc527
+                    cmykColor[1] = colToByte(cmyk.m);
3dc527
+                    cmykColor[2] = colToByte(cmyk.y);
3dc527
+                    cmykColor[3] = colToByte(cmyk.k);
3dc527
+
3dc527
+	            col[comp] |= cmykColor[comp];
3dc527
+	            hexBuf[i++] = cmykColor[comp];
3dc527
+	            if (i >= 64) {
3dc527
+	              writePSBuf(hexBuf, i);
3dc527
+	              i = 0;
3dc527
+	            }
3dc527
+	          }
3dc527
+	        } else {
3dc527
+	          // Gray color image
3dc527
+	          for (x = 0; x < w; ++x) {
3dc527
+	            if (likely(colorMode == splashModeRGB8)) {
3dc527
+                      inputColor.c[0] = byteToCol(p[3*x + 0]);
3dc527
+                      inputColor.c[1] = byteToCol(p[3*x + 1]);
3dc527
+                      inputColor.c[2] = byteToCol(p[3*x + 2]);
3dc527
+                    } else {
3dc527
+                      inputColor.c[0] = byteToCol(p[3*x + 2]);
3dc527
+                      inputColor.c[1] = byteToCol(p[3*x + 1]);
3dc527
+                      inputColor.c[2] = byteToCol(p[3*x + 0]);
3dc527
+                    }
3dc527
+                    rgbCS->getCMYK(&inputColor, &cmyk);
3dc527
+                    cmykColor[0] = colToByte(cmyk.c);
3dc527
+                    cmykColor[1] = colToByte(cmyk.m);
3dc527
+                    cmykColor[2] = colToByte(cmyk.y);
3dc527
+                    cmykColor[3] = colToByte(cmyk.k);
3dc527
+
3dc527
+	            col[comp] |= cmykColor[comp];
3dc527
+	            digit = cmykColor[comp] / 16;
3dc527
+	            hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
+	            digit = cmykColor[comp] % 16;
3dc527
+	            hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
+	            if (i >= 64) {
3dc527
+	              hexBuf[i++] = '\n';
3dc527
+	              writePSBuf(hexBuf, i);
3dc527
+	              i = 0;
3dc527
+	            }
3dc527
+	          }
3dc527
+	        }
3dc527
+              }
3dc527
+              p -= bitmap->getRowSize();
3dc527
+            }
3dc527
+            delete rgbCS;
3dc527
+            break;
3dc527
+          default:
3dc527
+            for (y = 0; y < h; ++y) {
3dc527
+              for (comp = 0; comp < 4; ++comp) {
3dc527
+	        if (useBinary) {
3dc527
+	          // Binary color image
3dc527
+	          for (x = 0; x < w; ++x) {
3dc527
+	            col[comp] |= p[4*x + comp];
3dc527
+	            hexBuf[i++] = p[4*x + comp];
3dc527
+	            if (i >= 64) {
3dc527
+	              writePSBuf(hexBuf, i);
3dc527
+	              i = 0;
3dc527
+	            }
3dc527
+	          }
3dc527
+	        } else {
3dc527
+	          // Gray color image
3dc527
+	          for (x = 0; x < w; ++x) {
3dc527
+	            col[comp] |= p[4*x + comp];
3dc527
+	            digit = p[4*x + comp] / 16;
3dc527
+	            hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
+	            digit = p[4*x + comp] % 16;
3dc527
+	            hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
+	            if (i >= 64) {
3dc527
+	              hexBuf[i++] = '\n';
3dc527
+	              writePSBuf(hexBuf, i);
3dc527
+	              i = 0;
3dc527
+	            }
3dc527
+	          }
3dc527
+	        }
3dc527
+              }
3dc527
+              p -= bitmap->getRowSize();
3dc527
+            }
3dc527
+          break;
3dc527
+        }
3dc527
+      } else {
3dc527
+	// Color image, do not need to check color flags
3dc527
+        switch (colorMode) {
3dc527
+          case splashModeRGB8:
3dc527
+          case splashModeBGR8:
3dc527
+            rgbCS = new GfxDeviceRGBColorSpace();
3dc527
+            for (y = 0; y < h; ++y) {
3dc527
+              for (comp = 0; comp < 4; ++comp) {
3dc527
+	        if (useBinary) {
3dc527
+	          // Binary color image
3dc527
+	          for (x = 0; x < w; ++x) {
3dc527
+	            if (likely(colorMode == splashModeRGB8)) {
3dc527
+                      inputColor.c[0] = byteToCol(p[3*x + 0]);
3dc527
+                      inputColor.c[1] = byteToCol(p[3*x + 1]);
3dc527
+                      inputColor.c[2] = byteToCol(p[3*x + 2]);
3dc527
+                    } else {
3dc527
+                      inputColor.c[0] = byteToCol(p[3*x + 2]);
3dc527
+                      inputColor.c[1] = byteToCol(p[3*x + 1]);
3dc527
+                      inputColor.c[2] = byteToCol(p[3*x + 0]);
3dc527
+                    }
3dc527
+                    rgbCS->getCMYK(&inputColor, &cmyk);
3dc527
+                    cmykColor[0] = colToByte(cmyk.c);
3dc527
+                    cmykColor[1] = colToByte(cmyk.m);
3dc527
+                    cmykColor[2] = colToByte(cmyk.y);
3dc527
+                    cmykColor[3] = colToByte(cmyk.k);
3dc527
+
3dc527
+	            hexBuf[i++] = cmykColor[comp];
3dc527
+	            if (i >= 64) {
3dc527
+	              writePSBuf(hexBuf, i);
3dc527
+	              i = 0;
3dc527
+	            }
3dc527
+	          }
3dc527
+	        } else {
3dc527
+	          // Hex color image
3dc527
+	          for (x = 0; x < w; ++x) {
3dc527
+	            if (likely(colorMode == splashModeRGB8)) {
3dc527
+                      inputColor.c[0] = byteToCol(p[3*x + 0]);
3dc527
+                      inputColor.c[1] = byteToCol(p[3*x + 1]);
3dc527
+                      inputColor.c[2] = byteToCol(p[3*x + 2]);
3dc527
+                    } else {
3dc527
+                      inputColor.c[0] = byteToCol(p[3*x + 2]);
3dc527
+                      inputColor.c[1] = byteToCol(p[3*x + 1]);
3dc527
+                      inputColor.c[2] = byteToCol(p[3*x + 0]);
3dc527
+                    }
3dc527
+                    rgbCS->getCMYK(&inputColor, &cmyk);
3dc527
+                    cmykColor[0] = colToByte(cmyk.c);
3dc527
+                    cmykColor[1] = colToByte(cmyk.m);
3dc527
+                    cmykColor[2] = colToByte(cmyk.y);
3dc527
+                    cmykColor[3] = colToByte(cmyk.k);
3dc527
+
3dc527
+	            digit = cmykColor[comp] / 16;
3dc527
+	            hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
+	            digit = cmykColor[comp] % 16;
3dc527
+	            hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
+	            if (i >= 64) {
3dc527
+	              hexBuf[i++] = '\n';
3dc527
+	              writePSBuf(hexBuf, i);
3dc527
+	              i = 0;
3dc527
+	            }
3dc527
+	          }
3dc527
+	        }
3dc527
+              }
3dc527
+              p -= bitmap->getRowSize();
3dc527
+            }
3dc527
+            delete rgbCS;
3dc527
+            break;
3dc527
+          default:
3dc527
+            for (y = 0; y < h; ++y) {
3dc527
+              for (comp = 0; comp < 4; ++comp) {
3dc527
+	        if (useBinary) {
3dc527
+	          // Binary color image
3dc527
+	          for (x = 0; x < w; ++x) {
3dc527
+	            hexBuf[i++] = p[4*x + comp];
3dc527
+	            if (i >= 64) {
3dc527
+	              writePSBuf(hexBuf, i);
3dc527
+	              i = 0;
3dc527
+	            }
3dc527
+	          }
3dc527
+	        } else {
3dc527
+	          // Hex color image
3dc527
+	          for (x = 0; x < w; ++x) {
3dc527
+	            digit = p[4*x + comp] / 16;
3dc527
+	            hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
+	            digit = p[4*x + comp] % 16;
3dc527
+	            hexBuf[i++] = digit + ((digit >= 10)? 'a' - 10: '0');
3dc527
+	            if (i >= 64) {
3dc527
+	              hexBuf[i++] = '\n';
3dc527
+	              writePSBuf(hexBuf, i);
3dc527
+	              i = 0;
3dc527
+	            }
3dc527
+	          }
3dc527
+	        }
3dc527
+              }
3dc527
+              p -= bitmap->getRowSize();
3dc527
+            }
3dc527
+          break;
3dc527
         }
3dc527
       }
3dc527
       if (i != 0) {