LCOV - code coverage report
Current view: top level - alg/viewshed - cumulative.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 77 87 88.5 %
Date: 2025-01-18 12:42:00 Functions: 9 9 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  * (c) 2024 info@hobu.co
       3             :  *
       4             :  * SPDX-License-Identifier: MIT
       5             :  ****************************************************************************/
       6             : 
       7             : #include <algorithm>
       8             : #include <limits>
       9             : #include <thread>
      10             : 
      11             : #include "cpl_worker_thread_pool.h"
      12             : 
      13             : #include "combiner.h"
      14             : #include "cumulative.h"
      15             : #include "notifyqueue.h"
      16             : #include "util.h"
      17             : #include "viewshed_executor.h"
      18             : 
      19             : namespace gdal
      20             : {
      21             : namespace viewshed
      22             : {
      23             : 
      24             : /// Constructor
      25             : ///
      26             : /// @param opts  Options for viewshed generation.
      27           1 : Cumulative::Cumulative(const Options &opts) : m_opts(opts)
      28             : {
      29           1 : }
      30             : 
      31             : /// Destructor
      32             : ///
      33             : Cumulative::~Cumulative() = default;
      34             : 
      35             : /// Compute the cumulative viewshed of a raster band.
      36             : ///
      37             : /// @param srcFilename  Source filename.
      38             : /// @param pfnProgress  Pointer to the progress function. Can be null.
      39             : /// @param pProgressArg  Argument passed to the progress function
      40             : /// @return  True on success, false otherwise.
      41           1 : bool Cumulative::run(const std::string &srcFilename,
      42             :                      GDALProgressFunc pfnProgress, void *pProgressArg)
      43             : {
      44             :     // In cumulative mode, we run the executors in normal mode and want "1" where things
      45             :     // are visible.
      46           1 :     m_opts.outputMode = OutputMode::Normal;
      47           1 :     m_opts.visibleVal = 1;
      48             : 
      49             :     DatasetPtr srcDS(
      50           2 :         GDALDataset::FromHandle(GDALOpen(srcFilename.c_str(), GA_ReadOnly)));
      51           1 :     if (!srcDS)
      52             :     {
      53           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Unable open source file.");
      54           0 :         return false;
      55             :     }
      56             : 
      57           1 :     GDALRasterBand *pSrcBand = srcDS->GetRasterBand(1);
      58             : 
      59             :     // In cumulative mode, the output extent is always the entire source raster.
      60           1 :     m_extent.xStop = GDALGetRasterBandXSize(pSrcBand);
      61           1 :     m_extent.yStop = GDALGetRasterBandYSize(pSrcBand);
      62             : 
      63             :     // Make a bunch of observer locations based on the spacing and stick them on a queue
      64             :     // to be handled by viewshed executors.
      65          22 :     for (int x = 0; x < m_extent.xStop; x += m_opts.observerSpacing)
      66         609 :         for (int y = 0; y < m_extent.yStop; y += m_opts.observerSpacing)
      67         588 :             m_observerQueue.push({x, y});
      68           1 :     m_observerQueue.done();
      69             : 
      70             :     // Run executors.
      71           1 :     const int numThreads = m_opts.numJobs;
      72           1 :     std::atomic<bool> err = false;
      73           1 :     std::atomic<int> running = numThreads;
      74             :     Progress progress(pfnProgress, pProgressArg,
      75           2 :                       m_observerQueue.size() * m_extent.ySize());
      76           2 :     CPLWorkerThreadPool executorPool(numThreads);
      77           4 :     for (int i = 0; i < numThreads; ++i)
      78           3 :         executorPool.SubmitJob(
      79           3 :             [this, &srcFilename, &progress, &err, &running]
      80           3 :             { runExecutor(srcFilename, progress, err, running); });
      81             : 
      82             :     // Run combiners that create 8-bit sums of executor jobs.
      83           2 :     CPLWorkerThreadPool combinerPool(numThreads);
      84             :     std::vector<Combiner> combiners(numThreads,
      85           3 :                                     Combiner(m_datasetQueue, m_rollupQueue));
      86           4 :     for (Combiner &c : combiners)
      87           6 :         combinerPool.SubmitJob([&c] { c.run(); });
      88             : 
      89             :     // Run 32-bit rollup job that combines the 8-bit results from the combiners.
      90           3 :     std::thread sum([this] { rollupRasters(); });
      91             : 
      92             :     // When the combiner jobs are done, all the data is in the rollup queue.
      93           1 :     combinerPool.WaitCompletion();
      94           1 :     if (m_datasetQueue.isStopped())
      95           0 :         return false;
      96           1 :     m_rollupQueue.done();
      97             : 
      98             :     // Wait for finalBuf to be fully filled.
      99           1 :     sum.join();
     100             :     // The executors should exit naturally, but we wait here so that we don't outrun their
     101             :     // completion and exit with outstanding threads.
     102           1 :     executorPool.WaitCompletion();
     103             : 
     104             :     // Scale the data so that we can write an 8-bit raster output.
     105           1 :     scaleOutput();
     106           1 :     if (!writeOutput(createOutputDataset(*pSrcBand, m_opts, m_extent)))
     107             :     {
     108           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     109             :                  "Unable to write to output file.");
     110           0 :         return false;
     111             :     }
     112           1 :     progress.emit(1);
     113             : 
     114           1 :     return true;
     115             : }
     116             : 
     117             : /// Run an executor (single viewshed)
     118             : /// @param srcFilename  Source filename
     119             : /// @param progress  Progress supporting support.
     120             : /// @param err  Shared error flag.
     121             : /// @param running  Shared count of number of executors running.
     122           3 : void Cumulative::runExecutor(const std::string &srcFilename, Progress &progress,
     123             :                              std::atomic<bool> &err, std::atomic<int> &running)
     124             : {
     125           6 :     DatasetPtr srcDs(GDALDataset::Open(srcFilename.c_str(), GA_ReadOnly));
     126           3 :     if (!srcDs)
     127             :     {
     128           0 :         err = true;
     129             :     }
     130             :     else
     131             :     {
     132             :         Location loc;
     133         591 :         while (!err && m_observerQueue.pop(loc))
     134             :         {
     135             :             GDALDriver *memDriver =
     136         588 :                 GetGDALDriverManager()->GetDriverByName("MEM");
     137         588 :             DatasetPtr dstDs(memDriver ? memDriver->Create("", m_extent.xSize(),
     138             :                                                            m_extent.ySize(), 1,
     139             :                                                            GDT_Byte, nullptr)
     140        1764 :                                        : nullptr);
     141         588 :             if (!dstDs)
     142             :             {
     143           0 :                 err = true;
     144             :             }
     145             :             else
     146             :             {
     147             :                 ViewshedExecutor executor(
     148         588 :                     *srcDs->GetRasterBand(1), *dstDs->GetRasterBand(1), loc.x,
     149        1765 :                     loc.y, m_extent, m_extent, m_opts, progress);
     150         588 :                 err = !executor.run();
     151         587 :                 if (!err)
     152         587 :                     m_datasetQueue.push(std::move(dstDs));
     153             :             }
     154             :         }
     155             :     }
     156             : 
     157             :     // Job done. Set the output queue state.  If all the executor jobs have completed,
     158             :     // set the dataset output queue done.
     159           3 :     if (err)
     160           0 :         m_datasetQueue.stop();
     161             :     else
     162             :     {
     163           3 :         running--;
     164           3 :         if (!running)
     165           1 :             m_datasetQueue.done();
     166             :     }
     167           3 : }
     168             : 
     169             : // Add 8-bit rasters into the 32-bit raster buffer.
     170           1 : void Cumulative::rollupRasters()
     171             : {
     172           1 :     DatasetPtr pDS;
     173             : 
     174           1 :     m_finalBuf.resize(m_extent.size());
     175           4 :     while (m_rollupQueue.pop(pDS))
     176             :     {
     177             :         uint8_t *srcP =
     178           3 :             static_cast<uint8_t *>(pDS->GetInternalHandle("MEMORY1"));
     179       43263 :         for (size_t i = 0; i < m_extent.size(); ++i)
     180       43260 :             m_finalBuf[i] += srcP[i];
     181             :     }
     182           1 : }
     183             : 
     184             : /// Scale the output so that it's fully spread in 8 bits. Perhaps this shouldn't happen if
     185             : /// the max is less than 255?
     186           1 : void Cumulative::scaleOutput()
     187             : {
     188           1 :     uint32_t m = 0;  // This gathers all the bits set.
     189       14421 :     for (uint32_t &val : m_finalBuf)
     190       14420 :         m = std::max(val, m);
     191             : 
     192           1 :     if (m == 0)
     193           0 :         return;
     194             : 
     195             :     double factor =
     196           1 :         std::numeric_limits<uint8_t>::max() / static_cast<double>(m);
     197       14421 :     for (uint32_t &val : m_finalBuf)
     198       14420 :         val = static_cast<uint32_t>(std::floor(factor * val));
     199             : }
     200             : 
     201             : /// Write the output dataset.
     202             : /// @param pDstDS  Pointer to the destination dataset.
     203             : /// @return True if the write was successful, false otherwise.
     204           1 : bool Cumulative::writeOutput(DatasetPtr pDstDS)
     205             : {
     206           1 :     if (!pDstDS)
     207           0 :         return false;
     208             : 
     209           1 :     GDALRasterBand *pDstBand = pDstDS->GetRasterBand(1);
     210           2 :     return (pDstBand->RasterIO(GF_Write, 0, 0, m_extent.xSize(),
     211           1 :                                m_extent.ySize(), m_finalBuf.data(),
     212             :                                m_extent.xSize(), m_extent.ySize(), GDT_UInt32,
     213           1 :                                0, 0, nullptr) == 0);
     214             : }
     215             : 
     216             : }  // namespace viewshed
     217             : }  // namespace gdal

Generated by: LCOV version 1.14