ExifTool Forum

ExifTool => The "exiftool" Application => Topic started by: xpsd300 on April 22, 2019, 05:59:05 PM

Title: JPEG Start of Scan (SOS) Marker
Post by: xpsd300 on April 22, 2019, 05:59:05 PM
Hi,

A publisher is asking us to provide the JPEG compression ratio for our photos.

From what I've read, this is a ratio of the maximum uncompressed size (Photoshop pixel dimensions) to the compressed file size.

I'm using the following calculation to get the maximum uncompressed size:

((document width * DPI) * (document height * DPI) * (channels * bits)) / 8

So for a 2 x 2.5 in. printed image at 300 DPI (24-bit), this computes to approx. 1.29 MB.

For the compressed size, I'm using the file size (not size on disk). For example, 200 KB.

Using these two values, I'm getting a ratio of approx. 6.6:1

Our images contain a lot a metadata for version control, so I know that this is skewing the values.

Is there a way with Exiftool to extract just the compressed image data from a file, or to identify where the SOS marker begins?

Thanks.
Title: Re: JPEG Start of Scan (SOS) Marker
Post by: StarGeek on April 22, 2019, 06:53:58 PM
I don't believe there's any direct way to get that info with exiftool.

You could use a temp file and do
exiftool -o Temp.jpg -all Input.jpg
which would create a temp file stripped of metadata, and then get the filesize of Temp.jpg.

Or, if you're on Unix (mac?), you could use
exiftool -o - -all= y:\!temp\Test4.jpg | wc -c
and the result would be the number of bytes (I think, not a unix command I've ever used before).

Off hand I can't find a pure Windows command that would work.  There is the Powershell Measure-Object -Character but since powershell will corrupt a binary pipe by converting it to ansi, you'd probably lose characters and get an inaccurate reading.
Title: Re: JPEG Start of Scan (SOS) Marker
Post by: StarGeek on April 22, 2019, 07:14:20 PM
You might take a look at JpegSnoop (https://www.impulseadventure.com/photo/jpeg-snoop.html).  It normally runs as a gui, but you can run it on the command line with something like
jpegsnoop -i Input.jpg  -scan -o Output.txt

Title: Re: JPEG Start of Scan (SOS) Marker
Post by: Phil Harvey on April 23, 2019, 07:31:52 AM
Using the position of the SOS marker won't give you the whole picture because there is often metadata after the EOI (MPF metadata for example).  Also, I would include some other required JPEG segments (like SOF, DHT, etc) in the image size.  It would probably be best just to just drop the APP segments and anything after the EOI from the calculation.  I could maybe add a JPEG "MetadataSize" tag to help you here.  Let me think about this.

- Phil

Edit: No.  Adding this tag would slow down processing because ExifTool doesn't read to the EOI unless necessary, so information about the trailer size isn't generally known to ExifTool.

Edit2: ...or I could only calculate this tag if requested.  I'll try adding a JPEGImageLength tag like this in ExifTool 11.38.
Title: Re: JPEG Start of Scan (SOS) Marker
Post by: xpsd300 on April 23, 2019, 07:49:06 PM
@StarGeek - Thanks for pointing out JpegSnoop. I get the following when I run a scan on one of our images:

*** Decoding SCAN Data ***
  OFFSET: 0x00007BF9
  Scan Decode Mode: No IDCT (DC only)
    NOTE: Low-resolution DC component shown. Can decode full-res with [Options->Scan Segment->Full IDCT]

  Scan Data encountered marker   0xFFD9 @ 0x0003E589.0

  Compression stats:
    Compression Ratio:  6.04:1
    Bits per pixel:     3.98:1



@Phil - From this marker, can you tell how the compression ratio is calculated? That appears to be exactly what the publisher is looking for. Is it something you'd be able to add to ExifTool?

The CSV output option of your tool has been a tremendous help to the 1000 images we're working on. Thanks.
Title: Re: JPEG Start of Scan (SOS) Marker
Post by: StarGeek on April 23, 2019, 08:32:23 PM
Looking at the source code (https://github.com/ImpulseAdventure/JPEGsnoop/) for JpegSnoop, I can find this bit in ImgDecode.cpp, in case that might help out deciding if it's the data you want.  It's a bit beyond me, but it seems to me that it doesn't include the dpi calculation in your first post. 

// Report Compression stats
// TODO: Should we use m_nNumSofComps?
strTmp.Format(_T("  Compression stats:"));
m_pLog->AddLine(strTmp);
float nCompressionRatio = (float)(m_nDimX*m_nDimY*m_nNumSosComps*8) / (float)((m_anScanBuffPtr_pos[0]-m_nScanBuffPtr_first)*8);
strTmp.Format(_T("    Compression Ratio: %5.2f:1"),nCompressionRatio);
Title: Re: JPEG Start of Scan (SOS) Marker
Post by: xpsd300 on April 23, 2019, 09:23:07 PM
Thanks for posting the source code. Looking before and after that snippet, and looking at the -scan output, it appears that the m_nDimX and m_nDimY values are in pixels.

The height and width dimensions I posted before are the "document size" values in Photoshop which appear to be equivalent when multiplied by DPI. For example:

Image Size Width (px) = Document Size Width (in) * DPI
Image Size Height (px) = Document Size Height (in) * DPI
Title: Re: JPEG Start of Scan (SOS) Marker
Post by: Phil Harvey on April 24, 2019, 10:21:31 AM
With ExifTool 11.38 (just released), the attached config file will generate a Composite JPEGCompression tag for you, with a command like this:

exiftool -config jpeg_compression.config -jpegcompression FILE

- Phil
Title: Re: JPEG Start of Scan (SOS) Marker
Post by: xpsd300 on April 24, 2019, 02:52:17 PM
@Phil - Thanks so much for doing this so quickly. It works great!

I've expanded the config file to also get the Bits Per Pixel:

%Image::ExifTool::UserDefined = (
    'Image::ExifTool::Composite' => {
        JPEGCompression => {
            Require => {
                0 => 'JPEGImageLength',
                1 => 'ImageWidth',
                2 => 'ImageHeight',
                3 => 'BitsPerSample',
                4 => 'ColorComponents',
            },
            ValueConv => '(($val[1]*$val[2]*$val[3]*$val[4])/8) / $val[0]',
            PrintConv => 'sprintf("%.2f:1",$val)',
        },

        JPEGBitsPerPixel => {
            Require => {
                0 => 'JPEGImageLength',
                1 => 'ImageWidth',
                2 => 'ImageHeight',
            },
            ValueConv => '$val[0] / (($val[1]*$val[2])/8)',
            PrintConv => 'sprintf("%.2f:1",$val)',
        },
    },
);

%Image::ExifTool::UserDefined::Options = (
    RequestTags => 'JPEGImageLength'
);

1; #end
Title: Re: JPEG Start of Scan (SOS) Marker
Post by: Phil Harvey on April 24, 2019, 03:17:42 PM
Nice.  Happy to have helped.

- Phil