LCOV - code coverage report
Current view: top level - frmts/jpeg - jpegdrivercore.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 52 62 83.9 %
Date: 2025-01-18 12:42:00 Functions: 3 3 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  JPEG JFIF Driver
       4             :  * Purpose:  Implement GDAL JPEG Support based on IJG libjpeg.
       5             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2000, Frank Warmerdam
       9             :  * Copyright (c) 2007-2014, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * Portions Copyright (c) Her majesty the Queen in right of Canada as
      12             :  * represented by the Minister of National Defence, 2006.
      13             :  *
      14             :  * SPDX-License-Identifier: MIT
      15             :  ****************************************************************************/
      16             : 
      17             : #include "jpegdrivercore.h"
      18             : 
      19             : // So that D_LOSSLESS_SUPPORTED is visible if defined in jmorecfg of libjpeg-turbo >= 2.2
      20             : #define JPEG_INTERNAL_OPTIONS
      21             : #include "jpeglib.h"
      22             : 
      23             : /************************************************************************/
      24             : /*                    JPEGDatasetIsJPEGLS()                             */
      25             : /************************************************************************/
      26             : 
      27        6639 : bool JPEGDatasetIsJPEGLS(GDALOpenInfo *poOpenInfo)
      28             : 
      29             : {
      30        6639 :     GByte *pabyHeader = poOpenInfo->pabyHeader;
      31        6639 :     int nHeaderBytes = poOpenInfo->nHeaderBytes;
      32             : 
      33        6639 :     if (nHeaderBytes < 10)
      34           0 :         return false;
      35             : 
      36        6639 :     if (pabyHeader[0] != 0xff || pabyHeader[1] != 0xd8)
      37           0 :         return false;
      38             : 
      39       30521 :     for (int nOffset = 2; nOffset + 4 < nHeaderBytes;)
      40             :     {
      41       26083 :         if (pabyHeader[nOffset] != 0xFF)
      42           6 :             return false;
      43             : 
      44       26077 :         int nMarker = pabyHeader[nOffset + 1];
      45       26077 :         if (nMarker == 0xDA)
      46        2195 :             return false;
      47             : 
      48       23882 :         if (nMarker == 0xF7)  // JPEG Extension 7, JPEG-LS.
      49           0 :             return true;
      50       23882 :         if (nMarker == 0xF8)  // JPEG Extension 8, JPEG-LS Extension.
      51           0 :             return true;
      52       23882 :         if (nMarker == 0xC3)  // Start of Frame 3 (Lossless Huffman)
      53           0 :             return true;
      54       23882 :         if (nMarker ==
      55             :             0xC7)  // Start of Frame 7 (Differential Lossless Huffman)
      56           0 :             return true;
      57       23882 :         if (nMarker == 0xCB)  // Start of Frame 11 (Lossless Arithmetic)
      58           0 :             return true;
      59       23882 :         if (nMarker ==
      60             :             0xCF)  // Start of Frame 15 (Differential Lossless Arithmetic)
      61           0 :             return true;
      62             : 
      63       23882 :         nOffset += 2 + pabyHeader[nOffset + 2] * 256 + pabyHeader[nOffset + 3];
      64             :     }
      65             : 
      66        4438 :     return false;
      67             : }
      68             : 
      69             : /************************************************************************/
      70             : /*                     JPEGDriverIdentify()                             */
      71             : /************************************************************************/
      72             : 
      73       59371 : int JPEGDriverIdentify(GDALOpenInfo *poOpenInfo)
      74             : 
      75             : {
      76             :     // If it is a subfile, read the JPEG header.
      77       59371 :     if (STARTS_WITH_CI(poOpenInfo->pszFilename, "JPEG_SUBFILE:"))
      78         300 :         return TRUE;
      79       59071 :     if (STARTS_WITH(poOpenInfo->pszFilename, "JPEG:"))
      80          10 :         return TRUE;
      81             : 
      82             :     // First we check to see if the file has the expected header bytes.
      83       59061 :     const int nHeaderBytes = poOpenInfo->nHeaderBytes;
      84             : 
      85       59061 :     if (nHeaderBytes < 10)
      86       48589 :         return FALSE;
      87             : 
      88       10472 :     GByte *const pabyHeader = poOpenInfo->pabyHeader;
      89       10472 :     if (pabyHeader[0] != 0xff || pabyHeader[1] != 0xd8 || pabyHeader[2] != 0xff)
      90        3833 :         return FALSE;
      91             : 
      92             :         // libjpeg-turbo >= 2.2 supports lossless mode
      93             : #if !defined(D_LOSSLESS_SUPPORTED)
      94        6639 :     if (JPEGDatasetIsJPEGLS(poOpenInfo))
      95             :     {
      96           0 :         return FALSE;
      97             :     }
      98             : #endif
      99             : 
     100             :     // Some files like
     101             :     // http://dionecanali.hd.free.fr/~mdione/mapzen/N65E039.hgt.gz could be
     102             :     // mis-identfied as JPEG
     103       13278 :     CPLString osFilenameLower = CPLString(poOpenInfo->pszFilename).tolower();
     104       13278 :     if (osFilenameLower.endsWith(".hgt") ||
     105       26556 :         osFilenameLower.endsWith(".hgt.gz") ||
     106       13278 :         osFilenameLower.endsWith(".hgt.zip"))
     107             :     {
     108           0 :         return FALSE;
     109             :     }
     110             : 
     111        6639 :     return TRUE;
     112             : }
     113             : 
     114             : /************************************************************************/
     115             : /*                     JPEGDriverSetCommonMetadata()                    */
     116             : /************************************************************************/
     117             : 
     118        1381 : void JPEGDriverSetCommonMetadata(GDALDriver *poDriver)
     119             : {
     120        1381 :     poDriver->SetDescription(DRIVER_NAME);
     121        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
     122        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "JPEG JFIF");
     123        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/jpeg.html");
     124        1381 :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "jpg");
     125        1381 :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSIONS, "jpg jpeg");
     126        1381 :     poDriver->SetMetadataItem(GDAL_DMD_MIMETYPE, "image/jpeg");
     127             : 
     128             : #if defined(JPEG_LIB_MK1_OR_12BIT) || defined(JPEG_DUAL_MODE_8_12)
     129        1381 :     poDriver->SetMetadataItem(GDAL_DMD_CREATIONDATATYPES, "Byte UInt16");
     130             : #else
     131             :     poDriver->SetMetadataItem(GDAL_DMD_CREATIONDATATYPES, "Byte");
     132             : #endif
     133        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
     134             : 
     135        1381 :     const char *pszOpenOptions =
     136             :         "<OpenOptionList>\n"
     137             :         "   <Option name='USE_INTERNAL_OVERVIEWS' type='boolean' "
     138             :         "description='whether to use implicit internal overviews' "
     139             :         "default='YES'/>\n"
     140             :         "   <Option name='APPLY_ORIENTATION' type='boolean' "
     141             :         "description='whether to take into account EXIF Orientation to "
     142             :         "rotate/flip the image' default='NO'/>\n"
     143             :         "</OpenOptionList>\n";
     144        1381 :     poDriver->SetMetadataItem(GDAL_DMD_OPENOPTIONLIST, pszOpenOptions);
     145             : 
     146             : #ifdef D_LOSSLESS_SUPPORTED
     147             :     // For autotest purposes
     148             :     poDriver->SetMetadataItem("LOSSLESS_JPEG_SUPPORTED", "YES", "JPEG");
     149             : #endif
     150             : 
     151        1381 :     poDriver->pfnIdentify = JPEGDriverIdentify;
     152        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_OPEN, "YES");
     153        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_CREATECOPY, "YES");
     154        1381 : }
     155             : 
     156             : /************************************************************************/
     157             : /*                    DeclareDeferredJPEGPlugin()                       */
     158             : /************************************************************************/
     159             : 
     160             : #ifdef PLUGIN_FILENAME
     161             : void DeclareDeferredJPEGPlugin()
     162             : {
     163             :     if (GDALGetDriverByName(DRIVER_NAME) != nullptr)
     164             :     {
     165             :         return;
     166             :     }
     167             :     auto poDriver = new GDALPluginDriverProxy(PLUGIN_FILENAME);
     168             : #ifdef PLUGIN_INSTALLATION_MESSAGE
     169             :     poDriver->SetMetadataItem(GDAL_DMD_PLUGIN_INSTALLATION_MESSAGE,
     170             :                               PLUGIN_INSTALLATION_MESSAGE);
     171             : #endif
     172             :     JPEGDriverSetCommonMetadata(poDriver);
     173             :     GetGDALDriverManager()->DeclareDeferredPluginDriver(poDriver);
     174             : }
     175             : #endif

Generated by: LCOV version 1.14