diff --git a/autotest/gdrivers/vrtprocesseddataset.py b/autotest/gdrivers/vrtprocesseddataset.py index fe076b8b8b08..e3ff9a1bf787 100755 --- a/autotest/gdrivers/vrtprocesseddataset.py +++ b/autotest/gdrivers/vrtprocesseddataset.py @@ -123,10 +123,13 @@ def test_vrtprocesseddataset_errors(tmp_vsimem): # Test nominal cases of BandAffineCombination algorithm -def test_vrtprocesseddataset_affine_combination_nominal(tmp_vsimem): +@pytest.mark.parametrize("INTERLEAVE", ["PIXEL", "BAND"]) +def test_vrtprocesseddataset_affine_combination_nominal(tmp_vsimem, INTERLEAVE): src_filename = str(tmp_vsimem / "src.tif") - src_ds = gdal.GetDriverByName("GTiff").Create(src_filename, 2, 1, 3) + src_ds = gdal.GetDriverByName("GTiff").Create( + src_filename, 2, 1, 3, options=["INTERLEAVE=" + INTERLEAVE] + ) src_ds.GetRasterBand(1).WriteArray(np.array([[1, 3]])) src_ds.GetRasterBand(2).WriteArray(np.array([[2, 6]])) src_ds.GetRasterBand(3).WriteArray(np.array([[3, 3]])) diff --git a/frmts/vrt/vrtprocesseddataset.cpp b/frmts/vrt/vrtprocesseddataset.cpp index 27ee1a437f69..d8b0875618a4 100644 --- a/frmts/vrt/vrtprocesseddataset.cpp +++ b/frmts/vrt/vrtprocesseddataset.cpp @@ -1040,32 +1040,91 @@ bool VRTProcessedDataset::ProcessRegion(int nXOff, int nYOff, int nBufXSize, CPLAssert(!m_aoSteps.empty()); + const size_t nPixelCount = static_cast(nBufXSize) * nBufYSize; + const int nFirstBandCount = m_aoSteps.front().nInBands; CPLAssert(nFirstBandCount == m_poSrcDS->GetRasterCount()); const GDALDataType eFirstDT = m_aoSteps.front().eInDT; const int nFirstDTSize = GDALGetDataTypeSizeBytes(eFirstDT); auto &abyInput = m_abyInput; auto &abyOutput = m_abyOutput; - try + + const char *pszInterleave = + m_poSrcDS->GetMetadataItem("INTERLEAVE", "IMAGE_STRUCTURE"); + if (nFirstBandCount > 1 && (!pszInterleave || EQUAL(pszInterleave, "BAND"))) { - abyInput.resize(static_cast(nBufXSize) * nBufYSize * - nFirstBandCount * nFirstDTSize); + // If there are several bands and the source dataset organization + // is apparently band interleaved, then first acquire data in + // a BSQ organization in the abyInput array use in the native + // data type. + // And then transpose it and convert it to the expected data type + // of the first step. + const auto eSrcDT = m_poSrcDS->GetRasterBand(1)->GetRasterDataType(); + try + { + abyInput.resize(nPixelCount * nFirstBandCount * + GDALGetDataTypeSizeBytes(eSrcDT)); + } + catch (const std::bad_alloc &) + { + CPLError(CE_Failure, CPLE_OutOfMemory, + "Out of memory allocating working buffer"); + return false; + } + + try + { + abyOutput.resize(nPixelCount * nFirstBandCount * nFirstDTSize); + } + catch (const std::bad_alloc &) + { + CPLError(CE_Failure, CPLE_OutOfMemory, + "Out of memory allocating working buffer"); + return false; + } + + CPLDebugOnly("VRT", "ProcessRegion(): start RasterIO()"); + if (m_poSrcDS->RasterIO(GF_Read, nXOff, nYOff, nBufXSize, nBufYSize, + abyInput.data(), nBufXSize, nBufYSize, eSrcDT, + nFirstBandCount, nullptr, 0, 0, 0, + nullptr) != CE_None) + { + return false; + } + CPLDebugOnly("VRT", "ProcessRegion(): end RasterIO()"); + + CPLDebugOnly("VRT", "ProcessRegion(): start GDALTranspose2D()"); + GDALTranspose2D(abyInput.data(), eSrcDT, abyOutput.data(), eFirstDT, + static_cast(nBufXSize) * nBufYSize, + nFirstBandCount); + CPLDebugOnly("VRT", "ProcessRegion(): end GDALTranspose2D()"); + + // Swap arrays + std::swap(abyInput, abyOutput); } - catch (const std::bad_alloc &) + else { - CPLError(CE_Failure, CPLE_OutOfMemory, - "Out of memory allocating working buffer"); - return false; - } + try + { + abyInput.resize(nPixelCount * nFirstBandCount * nFirstDTSize); + } + catch (const std::bad_alloc &) + { + CPLError(CE_Failure, CPLE_OutOfMemory, + "Out of memory allocating working buffer"); + return false; + } - if (m_poSrcDS->RasterIO( - GF_Read, nXOff, nYOff, nBufXSize, nBufYSize, abyInput.data(), - nBufXSize, nBufYSize, eFirstDT, nFirstBandCount, nullptr, - static_cast(nFirstDTSize) * nFirstBandCount, - static_cast(nFirstDTSize) * nFirstBandCount * nBufXSize, - nFirstDTSize, nullptr) != CE_None) - { - return false; + if (m_poSrcDS->RasterIO( + GF_Read, nXOff, nYOff, nBufXSize, nBufYSize, abyInput.data(), + nBufXSize, nBufYSize, eFirstDT, nFirstBandCount, nullptr, + static_cast(nFirstDTSize) * nFirstBandCount, + static_cast(nFirstDTSize) * nFirstBandCount * + nBufXSize, + nFirstDTSize, nullptr) != CE_None) + { + return false; + } } const double dfSrcXOff = nXOff; @@ -1096,8 +1155,7 @@ bool VRTProcessedDataset::ProcessRegion(int nXOff, int nYOff, int nBufXSize, { try { - abyOutput.resize(static_cast(nBufXSize) * nBufYSize * - oStep.nInBands * + abyOutput.resize(nPixelCount * oStep.nInBands * GDALGetDataTypeSizeBytes(oStep.eInDT)); } catch (const std::bad_alloc &) @@ -1110,16 +1168,14 @@ bool VRTProcessedDataset::ProcessRegion(int nXOff, int nYOff, int nBufXSize, GDALCopyWords64(abyInput.data(), eLastDT, GDALGetDataTypeSizeBytes(eLastDT), abyOutput.data(), oStep.eInDT, GDALGetDataTypeSizeBytes(oStep.eInDT), - static_cast(nBufXSize) * nBufYSize * - oStep.nInBands); + nPixelCount * oStep.nInBands); std::swap(abyInput, abyOutput); } try { - abyOutput.resize(static_cast(nBufXSize) * nBufYSize * - oStep.nOutBands * + abyOutput.resize(nPixelCount * oStep.nOutBands * GDALGetDataTypeSizeBytes(oStep.eOutDT)); } catch (const std::bad_alloc &)