Skip to content

Commit 6773dd0

Browse files
authored
Merge pull request OSGeo#11857 from rouault/fix_11851
GTiff: CreateCopy(): fix incorrect writing of mask band when COPY_SRC_OVERVIEWS=YES and INTERLEAVE=BAND (master only)
2 parents 4121bb8 + 3960fc8 commit 6773dd0

File tree

2 files changed

+33
-55
lines changed

2 files changed

+33
-55
lines changed

autotest/gcore/cog.py

+23-48
Original file line numberDiff line numberDiff line change
@@ -1996,46 +1996,19 @@ def test_cog_preserve_ALPHA_PREMULTIPLIED_on_copy(tmp_vsimem):
19961996

19971997

19981998
@gdaltest.enable_exceptions()
1999-
def test_cog_write_interleave_tile(tmp_vsimem):
1999+
@pytest.mark.parametrize("INTERLEAVE", ["TILE", "BAND"])
2000+
def test_cog_write_interleave_tile_or_band(tmp_vsimem, INTERLEAVE):
20002001
out_filename = str(tmp_vsimem / "out.tif")
20012002

20022003
with gdal.quiet_errors():
20032004
gdal.GetDriverByName("COG").CreateCopy(
20042005
out_filename,
20052006
gdal.Open("data/rgbsmall.tif"),
2006-
options=["INTERLEAVE=TILE", "BLOCKSIZE=32"],
2007+
options=["INTERLEAVE=" + INTERLEAVE, "BLOCKSIZE=32"],
20072008
)
20082009

20092010
ds = gdal.Open(out_filename)
2010-
assert ds.GetMetadataItem("INTERLEAVE", "IMAGE_STRUCTURE") == "TILE"
2011-
assert ds.GetMetadataItem("LAYOUT", "IMAGE_STRUCTURE") == "COG"
2012-
2013-
assert [ds.GetRasterBand(band + 1).Checksum() for band in range(3)] == [
2014-
21212,
2015-
21053,
2016-
21349,
2017-
]
2018-
2019-
_check_cog(out_filename)
2020-
2021-
2022-
###############################################################################
2023-
#
2024-
2025-
2026-
@gdaltest.enable_exceptions()
2027-
def test_cog_write_interleave_band(tmp_vsimem):
2028-
out_filename = str(tmp_vsimem / "out.tif")
2029-
2030-
with gdal.quiet_errors():
2031-
gdal.GetDriverByName("COG").CreateCopy(
2032-
out_filename,
2033-
gdal.Open("data/rgbsmall.tif"),
2034-
options=["INTERLEAVE=BAND", "BLOCKSIZE=32"],
2035-
)
2036-
2037-
ds = gdal.Open(out_filename)
2038-
assert ds.GetMetadataItem("INTERLEAVE", "IMAGE_STRUCTURE") == "BAND"
2011+
assert ds.GetMetadataItem("INTERLEAVE", "IMAGE_STRUCTURE") == INTERLEAVE
20392012
assert ds.GetMetadataItem("LAYOUT", "IMAGE_STRUCTURE") == "COG"
20402013

20412014
assert [ds.GetRasterBand(band + 1).Checksum() for band in range(3)] == [
@@ -2052,7 +2025,8 @@ def test_cog_write_interleave_band(tmp_vsimem):
20522025

20532026

20542027
@gdaltest.enable_exceptions()
2055-
def test_cog_write_interleave_tile_with_mask(tmp_vsimem):
2028+
@pytest.mark.parametrize("INTERLEAVE", ["TILE", "BAND"])
2029+
def test_cog_write_interleave_with_mask(tmp_vsimem, INTERLEAVE):
20562030
out_filename = str(tmp_vsimem / "out.tif")
20572031

20582032
with gdal.quiet_errors():
@@ -2061,11 +2035,11 @@ def test_cog_write_interleave_tile_with_mask(tmp_vsimem):
20612035
gdal.Translate(
20622036
"", "data/stefan_full_rgba.tif", options="-f MEM -b 1 -b 2 -b 3 -mask 4"
20632037
),
2064-
options=["INTERLEAVE=TILE", "BLOCKSIZE=32"],
2038+
options=["INTERLEAVE=" + INTERLEAVE, "BLOCKSIZE=32"],
20652039
)
20662040

20672041
ds = gdal.Open(out_filename)
2068-
assert ds.GetMetadataItem("INTERLEAVE", "IMAGE_STRUCTURE") == "TILE"
2042+
assert ds.GetMetadataItem("INTERLEAVE", "IMAGE_STRUCTURE") == INTERLEAVE
20692043
assert ds.GetMetadataItem("LAYOUT", "IMAGE_STRUCTURE") == "COG"
20702044

20712045
assert [ds.GetRasterBand(band + 1).Checksum() for band in range(3)] == [
@@ -2078,24 +2052,25 @@ def test_cog_write_interleave_tile_with_mask(tmp_vsimem):
20782052
_check_cog(out_filename)
20792053

20802054
# Check that the tiles are in the expected order in the file
2081-
last_offset = 0
2082-
for y in range(2):
2083-
for x in range(2):
2084-
for band in range(3):
2085-
offset = int(
2086-
ds.GetRasterBand(band + 1).GetMetadataItem(
2087-
f"BLOCK_OFFSET_{x}_{y}", "TIFF"
2055+
if INTERLEAVE == "TILE":
2056+
last_offset = 0
2057+
for y in range(2):
2058+
for x in range(2):
2059+
for band in range(3):
2060+
offset = int(
2061+
ds.GetRasterBand(band + 1).GetMetadataItem(
2062+
f"BLOCK_OFFSET_{x}_{y}", "TIFF"
2063+
)
20882064
)
2065+
assert offset > last_offset
2066+
last_offset = offset
2067+
offset = int(
2068+
ds.GetRasterBand(1)
2069+
.GetMaskBand()
2070+
.GetMetadataItem(f"BLOCK_OFFSET_{x}_{y}", "TIFF")
20892071
)
20902072
assert offset > last_offset
20912073
last_offset = offset
2092-
offset = int(
2093-
ds.GetRasterBand(1)
2094-
.GetMaskBand()
2095-
.GetMetadataItem(f"BLOCK_OFFSET_{x}_{y}", "TIFF")
2096-
)
2097-
assert offset > last_offset
2098-
last_offset = offset
20992074

21002075

21012076
###############################################################################

frmts/gtiff/gtiffdataset_write.cpp

+10-7
Original file line numberDiff line numberDiff line change
@@ -6700,18 +6700,21 @@ CPLErr GTiffDataset::CopyImageryAndMask(GTiffDataset *poDstDS,
67006700
{
67016701
memset(pBlockBuffer, 0,
67026702
static_cast<size_t>(poDstDS->m_nBlockXSize) *
6703-
poDstDS->m_nBlockYSize * nDataTypeSize);
6703+
poDstDS->m_nBlockYSize);
67046704
}
67056705
eErr = poSrcMaskBand->RasterIO(
67066706
GF_Read, iX, iY, nReqXSize, nReqYSize, pBlockBuffer,
6707-
nReqXSize, nReqYSize, eType, nDataTypeSize,
6708-
static_cast<GSpacing>(nDataTypeSize) *
6709-
poDstDS->m_nBlockXSize,
6710-
nullptr);
6707+
nReqXSize, nReqYSize, GDT_Byte, 1,
6708+
poDstDS->m_nBlockXSize, nullptr);
67116709
if (eErr == CE_None)
67126710
{
6713-
eErr = poDstDS->m_poMaskDS->WriteEncodedTileOrStrip(
6714-
iBlockMask, pBlockBuffer, false);
6711+
// Avoid any attempt to load from disk
6712+
poDstDS->m_poMaskDS->m_nLoadedBlock = iBlockMask;
6713+
eErr =
6714+
poDstDS->m_poMaskDS->GetRasterBand(1)->WriteBlock(
6715+
nXBlock, nYBlock, pBlockBuffer);
6716+
if (eErr == CE_None)
6717+
eErr = poDstDS->m_poMaskDS->FlushBlockBuf();
67156718
}
67166719

67176720
iBlockMask++;

0 commit comments

Comments
 (0)