LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/libkml - ogrlibkmldriver.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 48 57 84.2 %
Date: 2025-01-18 12:42:00 Functions: 5 5 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  KML Translator
       4             :  * Purpose:  Implements OGRLIBKMLDriver
       5             :  * Author:   Brian Case, rush at winkey dot org
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2010, Brian Case
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  *****************************************************************************/
      12             : 
      13             : #include "libkml_headers.h"
      14             : 
      15             : #include "ogr_libkml.h"
      16             : #include "ogrlibkmldrivercore.h"
      17             : #include "cpl_conv.h"
      18             : #include "cpl_error.h"
      19             : #include "cpl_multiproc.h"
      20             : 
      21             : using kmldom::KmlFactory;
      22             : 
      23             : static CPLMutex *hMutex = nullptr;
      24             : static KmlFactory *m_poKmlFactory = nullptr;
      25             : 
      26             : /******************************************************************************
      27             :  OGRLIBKMLDriverUnload()
      28             : ******************************************************************************/
      29             : 
      30          19 : static void OGRLIBKMLDriverUnload(GDALDriver * /* poDriver */)
      31             : {
      32          19 :     if (hMutex != nullptr)
      33          11 :         CPLDestroyMutex(hMutex);
      34          19 :     hMutex = nullptr;
      35          19 :     m_poKmlFactory = nullptr;
      36          19 : }
      37             : 
      38             : /******************************************************************************
      39             :  Open()
      40             : ******************************************************************************/
      41             : 
      42         288 : static GDALDataset *OGRLIBKMLDriverOpen(GDALOpenInfo *poOpenInfo)
      43             : {
      44         288 :     if (OGRLIBKMLDriverIdentify(poOpenInfo) == FALSE)
      45           0 :         return nullptr;
      46             : 
      47             :     {
      48         576 :         CPLMutexHolderD(&hMutex);
      49         288 :         if (m_poKmlFactory == nullptr)
      50          12 :             m_poKmlFactory = KmlFactory::GetFactory();
      51             :     }
      52             : 
      53         288 :     OGRLIBKMLDataSource *poDS = new OGRLIBKMLDataSource(m_poKmlFactory);
      54             : 
      55         288 :     if (!poDS->Open(poOpenInfo->pszFilename, poOpenInfo->eAccess == GA_Update))
      56             :     {
      57         162 :         delete poDS;
      58             : 
      59         162 :         poDS = nullptr;
      60             :     }
      61             : 
      62         288 :     return poDS;
      63             : }
      64             : 
      65             : /************************************************************************/
      66             : /*                               Create()                               */
      67             : /************************************************************************/
      68             : 
      69         103 : static GDALDataset *OGRLIBKMLDriverCreate(const char *pszName, int /* nBands */,
      70             :                                           int /* nXSize */, int /* nYSize */,
      71             :                                           GDALDataType /* eDT */,
      72             :                                           char **papszOptions)
      73             : {
      74         103 :     CPLAssert(nullptr != pszName);
      75         103 :     CPLDebug("LIBKML", "Attempt to create: %s", pszName);
      76             : 
      77             :     {
      78         206 :         CPLMutexHolderD(&hMutex);
      79         103 :         if (m_poKmlFactory == nullptr)
      80           1 :             m_poKmlFactory = KmlFactory::GetFactory();
      81             :     }
      82             : 
      83         103 :     OGRLIBKMLDataSource *poDS = new OGRLIBKMLDataSource(m_poKmlFactory);
      84             : 
      85         103 :     if (!poDS->Create(pszName, papszOptions))
      86             :     {
      87           1 :         delete poDS;
      88             : 
      89           1 :         poDS = nullptr;
      90             :     }
      91             : 
      92         103 :     return poDS;
      93             : }
      94             : 
      95             : /******************************************************************************
      96             :  DeleteDataSource()
      97             : 
      98             :  Note: This method recursively deletes an entire dir if the datasource is a dir
      99             :        and all the files are kml or kmz.
     100             : 
     101             : ******************************************************************************/
     102             : 
     103          67 : static CPLErr OGRLIBKMLDriverDelete(const char *pszName)
     104             : {
     105             :     /***** dir *****/
     106             :     VSIStatBufL sStatBuf;
     107          67 :     if (!VSIStatL(pszName, &sStatBuf) && VSI_ISDIR(sStatBuf.st_mode))
     108             :     {
     109          32 :         char **papszDirList = VSIReadDir(pszName);
     110          32 :         for (int iFile = 0;
     111          32 :              papszDirList != nullptr && papszDirList[iFile] != nullptr; iFile++)
     112             :         {
     113          32 :             if (CE_Failure == OGRLIBKMLDriverDelete(papszDirList[iFile]))
     114             :             {
     115          32 :                 CSLDestroy(papszDirList);
     116          32 :                 return CE_Failure;
     117             :             }
     118             :         }
     119           0 :         CSLDestroy(papszDirList);
     120             : 
     121           0 :         if (VSIRmdir(pszName) < 0)
     122             :         {
     123           0 :             return CE_Failure;
     124             :         }
     125             :     }
     126             : 
     127             :     /***** kml *****/
     128          35 :     else if (EQUAL(CPLGetExtensionSafe(pszName).c_str(), "kml"))
     129             :     {
     130          35 :         if (VSIUnlink(pszName) < 0)
     131          32 :             return CE_Failure;
     132             :     }
     133             : 
     134             :     /***** kmz *****/
     135           0 :     else if (EQUAL(CPLGetExtensionSafe(pszName).c_str(), "kmz"))
     136             :     {
     137           0 :         if (VSIUnlink(pszName) < 0)
     138           0 :             return CE_Failure;
     139             :     }
     140             : 
     141             :     /***** do not delete other types of files *****/
     142             :     else
     143             :     {
     144           0 :         return CE_Failure;
     145             :     }
     146             : 
     147             :     // TODO(schwehr): Isn't this redundant to the else case?
     148           3 :     return CE_None;
     149             : }
     150             : 
     151             : /******************************************************************************
     152             :  RegisterOGRLIBKML()
     153             : ******************************************************************************/
     154             : 
     155          24 : void RegisterOGRLIBKML()
     156             : {
     157          24 :     if (GDALGetDriverByName(DRIVER_NAME) != nullptr)
     158           0 :         return;
     159             : 
     160          24 :     GDALDriver *poDriver = new GDALDriver();
     161          24 :     OGRLIBKMLDriverSetCommonMetadata(poDriver);
     162             : 
     163          24 :     poDriver->pfnOpen = OGRLIBKMLDriverOpen;
     164          24 :     poDriver->pfnCreate = OGRLIBKMLDriverCreate;
     165          24 :     poDriver->pfnDelete = OGRLIBKMLDriverDelete;
     166          24 :     poDriver->pfnUnloadDriver = OGRLIBKMLDriverUnload;
     167             : 
     168          24 :     GetGDALDriverManager()->RegisterDriver(poDriver);
     169             : }

Generated by: LCOV version 1.14