From 8ef3a1143efb70513add81ef3fece0d1ed550075 Mon Sep 17 00:00:00 2001 From: William Silversmith Date: Tue, 21 May 2024 19:13:14 -0400 Subject: [PATCH] feat: voxel connectivity graph to labeled image (#123) * feat: add color connectivty graph for 4 and 6 connectivities * chore: whitespace * feat: add 6 and 8 color connectivity * feat: added colorization for 8-bit 8-connected * feat: add support for 8-connected * feat: add support for 26 connected vcg coloring * fix: start numbering from 1, iterate over z * test: check that 6-way color connectivity works * fix: handle different array sizes * fix: two recurring issues in 6 and 8 connected * test: check 4 and 8 connectivity for 4 connected logic * test: remove debugging statements * fix: mixed up uint8 and uint32 8-connected * fix: label incrementing for 6 and 26 connected * test: a basic test for 8-connected uint8 * test: basic test for 8-way uint32 * test: add check for 6-connectivity * test: 6-connected won't accept uint8 * fix: limit print out of DisjointSet to actual size * fix: incorrect limitation * fix: segmentation fault * fix: more segmentation faults * fix: segfault * test: check 26-connected supports 6-connected * test: check for simple 26-connectedness * test: check connecting center * perf: reduce memory usage for 26-connected * fix: segmentation fault * fix: segmentation fault * ensure self-contact is handled properly --- automated_test.py | 164 +++++++++++++ cc3d.hpp | 5 +- cc3d.pyx | 87 +++++++ cc3d_graphs.hpp | 598 +++++++++++++++++++++++++++++++++++++++++++++- 4 files changed, 851 insertions(+), 3 deletions(-) diff --git a/automated_test.py b/automated_test.py index de3ed34..eb70ae8 100644 --- a/automated_test.py +++ b/automated_test.py @@ -1260,3 +1260,167 @@ def test_periodic_boundary_6(): assert N == 2 +@pytest.mark.parametrize("dtype", [np.uint8, np.uint32]) +@pytest.mark.parametrize("connectivity", [4,8]) +def test_color_connectivity_graph_4(dtype, connectivity): + vcg = np.array([[]], dtype=dtype) + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + assert cc_labels.size == 0 + + vcg = np.zeros([2,1], dtype=dtype) + vcg[:] = 0x0f + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + ans = np.array([1,1], dtype=np.uint32) + assert np.all(cc_labels == ans) + + vcg = np.zeros([2,1], dtype=dtype) + vcg[:] = 0b1100 + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + ans = np.array([[1],[2]], dtype=np.uint32) + assert np.all(cc_labels == ans) + + vcg = np.zeros([1,2], dtype=dtype) + vcg[:] = 0b1100 + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + ans = np.array([1,1], dtype=np.uint32) + assert np.all(cc_labels == ans) + + vcg[:] = 0b0011 + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + + ans = np.array([1,2], dtype=np.uint32) + assert np.all(cc_labels == ans) + + vcg = np.zeros([2,2], dtype=dtype) + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + ans = np.array([[1, 3],[2, 4]], dtype=np.uint32) + assert np.all(cc_labels == ans) + + vcg[:] = 0b1111 + vcg[0,0] = 0b1110 + vcg[1,0] = 0b1101 + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + ans = np.array([[1, 1],[1, 1]], dtype=np.uint32) + assert np.all(cc_labels == ans) + + vcg[:] = 0b1111 + vcg[0,0] = 0b1110 + vcg[1,0] = 0b1101 + vcg[1,1] = 0b0111 + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + ans = np.array([[1, 1],[2, 1]], dtype=np.uint32) + assert np.all(cc_labels == ans) + + vcg[:] = 0b1111 + vcg[0,0] = 0b1010 + vcg[1,0] = 0b1101 + vcg[0,1] = 0b0111 + vcg[1,1] = 0b0111 + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=connectivity) + ans = np.array([[1, 3],[2, 3]], dtype=np.uint32) + assert np.all(cc_labels == ans) + +def test_color_connectivity_graph_8_uint8(): + vcg = np.zeros([2,2], dtype=np.uint8) + vcg[0,0] = 0b10000 + vcg[1,1] = 0b10000000 + vcg[1,0] = 0b100000 + vcg[0,1] = 0b1000000 + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=8) + ans = np.array([[1, 2],[2, 1]], dtype=np.uint32) + assert np.all(cc_labels == ans) + +def test_color_connectivity_graph_8_uint32(): + vcg = np.zeros([2,2], dtype=np.uint32) + vcg[0,0] = 0b1000000 + vcg[1,1] = 0b1000000000 + vcg[1,0] = 0b10000000 + vcg[0,1] = 0b100000000 + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=8) + ans = np.array([[1, 2],[2, 1]], dtype=np.uint32) + assert np.all(cc_labels == ans) + +def test_color_connectivity_graph_6(): + + with pytest.raises(ValueError): + vcg = np.zeros([2,2,2], dtype=np.uint8, order="F") + cc3d.color_connectivity_graph(vcg, connectivity=6) + + + vcg = np.zeros([10,10,10], dtype=np.uint32, order="F") + vcg[:,:,:] = 0b11111111111111111111111111 + + vcg[:,:,4] = vcg[:,:,4] & 0b101111 + vcg[:,:,5] = vcg[:,:,5] & 0b011111 + + vcg[:,4,:] = vcg[:,4,:] & 0b111011 + vcg[:,5,:] = vcg[:,5,:] & 0b110111 + + vcg[4,:,:] = vcg[4,:,:] & 0b111110 + vcg[5,:,:] = vcg[5,:,:] & 0b111101 + + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=6) + + out = np.zeros(vcg.shape, dtype=np.uint32, order="F") + out[:5,:5,:5] = 1 + out[5:,:5,:5] = 2 + out[:5,5:,:5] = 3 + out[5:,5:,:5] = 4 + out[:5,:5,5:] = 5 + out[5:,:5,5:] = 6 + out[:5,5:,5:] = 7 + out[5:,5:,5:] = 8 + + assert np.all(out == cc_labels) + +def test_color_connectivity_graph_26(): + + with pytest.raises(ValueError): + vcg = np.zeros([2,1,2], dtype=np.uint8, order="F") + cc3d.color_connectivity_graph(vcg, connectivity=26) + + vcg = np.zeros([10,10,10], dtype=np.uint32, order="F") + vcg[:,:,:] = 0b11111111111111111111111111 + + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=26) + assert np.all(cc_labels == 1) + + vcg[:,:,4] = vcg[:,:,4] & 0b101111 + vcg[:,:,5] = vcg[:,:,5] & 0b011111 + + vcg[:,4,:] = vcg[:,4,:] & 0b111011 + vcg[:,5,:] = vcg[:,5,:] & 0b110111 + + vcg[4,:,:] = vcg[4,:,:] & 0b111110 + vcg[5,:,:] = vcg[5,:,:] & 0b111101 + + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=26) + + out = np.zeros(vcg.shape, dtype=np.uint32, order="F") + out[:5,:5,:5] = 1 + out[5:,:5,:5] = 2 + out[:5,5:,:5] = 3 + out[5:,5:,:5] = 4 + out[:5,:5,5:] = 5 + out[5:,:5,5:] = 6 + out[:5,5:,5:] = 7 + out[5:,5:,5:] = 8 + + assert np.all(out == cc_labels) + + vcg[4,4,4] = 0b11111111111111111111000000 + vcg[5,4,4] = 0b11111111111111111111000000 + vcg[4,5,4] = 0b11111111111111111111000000 + vcg[5,5,4] = 0b11111111111111111111000000 + + vcg[4,4,5] = 0b11111111111111111111000000 + vcg[5,4,5] = 0b11111111111111111111000000 + vcg[4,5,5] = 0b11111111111111111111000000 + vcg[5,5,5] = 0b11111111111111111111000000 + + cc_labels = cc3d.color_connectivity_graph(vcg, connectivity=26) + assert np.all(cc_labels == 1) + + + + diff --git a/cc3d.hpp b/cc3d.hpp index 56d09f2..3c27f07 100644 --- a/cc3d.hpp +++ b/cc3d.hpp @@ -137,7 +137,8 @@ class DisjointSet { } void print() { - for (int i = 0; i < 15; i++) { + int size = std::min(static_cast(length), 15); + for (int i = 0; i < size; i++) { printf("%d, ", ids[i]); } printf("\n"); @@ -1082,7 +1083,7 @@ OUT* connected_components2d_8( else { next_label++; out_labels[loc] = next_label; - equivalences.add(out_labels[loc]); + equivalences.add(out_labels[loc]); } } } diff --git a/cc3d.pyx b/cc3d.pyx index 88ff769..96a7f4d 100644 --- a/cc3d.pyx +++ b/cc3d.pyx @@ -93,6 +93,13 @@ cdef extern from "cc3d_graphs.hpp" namespace "cc3d": vector[cpp_pair[size_t, size_t]] all_runs, T* labels, size_t voxels ) except + + cdef OUT* color_connectivity_graph_N[T,OUT]( + T* vcg, + int64_t sx, int64_t sy, int64_t sz, + int connectivity, + OUT* out_labels, + uint64_t& N + ) except + ctypedef fused UINT: uint8_t @@ -781,6 +788,86 @@ def _statistics_helper( return output +@cython.binding(True) +def color_connectivity_graph( + vcg, + connectivity = 26, + return_N = False, +) -> np.ndarray: + """ + Given a voxel connectivity graph following the same bit convention as + cc3d.voxel_connectivity_graph (see docstring), assuming an undirected + graph (the format supports directed graphs, but this is not implemented + for the sake of efficiency), this function will return a uint32 image + that contains connected components labeled according to the boundaries + described in the voxel connectivity graph (vcg). + """ + cdef int dims = len(vcg.shape) + if dims not in (2,3): + raise DimensionError("Only 2D, and 3D arrays supported. Got: " + str(dims)) + + if dims == 2 and connectivity not in [4,8,6,26]: + raise ValueError(f"Only 4 and 8 connectivity is supported for 2D images. Got: {connectivity}") + elif dims != 2 and connectivity not in [6,26]: + raise ValueError(f"Only 6 and 26 connectivity are supported for 3D images. Got: {connectivity}") + + dtype = vcg.dtype + if dtype not in [np.uint8, np.uint32]: + raise ValueError(f"Only uint8 and uint32 are supported. Got: {vcg.dtype}") + + if vcg.size == 0: + return np.zeros([0] * dims, dtype=np.uint32, order="F") + + while vcg.ndim < 3: + vcg = vcg[..., np.newaxis] + + vcg = np.asfortranarray(vcg) + + shape = vcg.shape + + cdef int sx = shape[0] + cdef int sy = shape[1] + cdef int sz = shape[2] + + if connectivity in [6, 26] and sz > 1 and dtype != np.uint32: + raise ValueError(f"Only uint32 is supported for 3d connectivites. Got: {vcg.dtype}") + + cdef uint8_t[:,:,:] arr_memview8u + cdef uint32_t[:,:,:] arr_memview32u + cdef uint32_t[:,:,:] out_labels32 + + cdef int64_t voxels = sx * sy * sz + out_labels = np.zeros( (sx,sy,sz,), dtype=np.uint32, order='F' ) + out_labels32 = out_labels + + cdef size_t N = 0 + + if dtype == np.uint32: + arr_memview32u = vcg + color_connectivity_graph_N[uint32_t, uint32_t]( + &arr_memview32u[0,0,0], + sx, sy, sz, + connectivity, + &out_labels32[0,0,0], + N + ) + else: + arr_memview8u = vcg + color_connectivity_graph_N[uint8_t, uint32_t]( + &arr_memview8u[0,0,0], + sx, sy, sz, + connectivity, + &out_labels32[0,0,0], + N + ) + + while out_labels.ndim > dims: + out_labels = out_labels[...,0] + + if return_N: + return (out_labels, N) + return out_labels + @cython.binding(True) def voxel_connectivity_graph( data:np.ndarray, diff --git a/cc3d_graphs.hpp b/cc3d_graphs.hpp index 591a551..982ba4c 100644 --- a/cc3d_graphs.hpp +++ b/cc3d_graphs.hpp @@ -1,6 +1,7 @@ #ifndef CC3D_GRAPHS_HPP #define CC3D_GRAPHS_HPP +#include #include #include #include @@ -10,6 +11,8 @@ #include #include +#include "cc3d.hpp" + namespace cc3d { // The voxel connectivity graph is specified as a directed graph @@ -209,7 +212,7 @@ OUT* extract_voxel_connectivity_graph( graph = extract_voxel_connectivity_graph_3d( in_labels, sx, sy, sz, graph ); - return and_mask(0b00111111, graph, sx, sy, sz); + return and_mask(0b00111111, graph, sx, sy, sz); } else if (connectivity == 8) { if (sz != 1) { @@ -406,6 +409,599 @@ void set_run_voxels( } } +template +OUT* simplified_relabel( + OUT* out_labels, const int64_t voxels, + const int64_t num_labels, DisjointSet &equivalences, + size_t &N = _dummy_N + ) { + + OUT label; + std::unique_ptr renumber(new OUT[num_labels + 1]()); + OUT next_label = 1; + + for (int64_t i = 1; i <= num_labels; i++) { + label = equivalences.root(i); + if (renumber[label] == 0) { + renumber[label] = next_label; + renumber[i] = next_label; + next_label++; + } + else { + renumber[i] = renumber[label]; + } + } + + // Raster Scan 2: Write final labels based on equivalences + N = next_label - 1; + for (int64_t loc = 0; loc < voxels; loc++) { + out_labels[loc] = renumber[out_labels[loc]]; + } + + return out_labels; +} + +template +OUT* color_connectivity_graph_26( + const uint8_t* vcg, // voxel connectivity graph + const int64_t sx, const int64_t sy, const int64_t sz, + OUT* out_labels = NULL, + size_t &N = _dummy_N +) { + throw new std::runtime_error("26-connectivity requires a 32-bit voxel graph for color_connectivity_graph_26."); +} + +template +uint64_t estimate_vcg_provisional_label_count( + T* vcg, const int64_t sx, const int64_t voxels +) { + uint64_t count = 0; // number of transitions between labels + + for (int64_t row = 0, loc = 0; loc < voxels; loc += sx, row++) { + count++; + for (int64_t x = 1; x < sx; x++) { + count += static_cast((vcg[loc + x] & 0b10) == 0); + } + } + + return count; +} + +template +OUT* color_connectivity_graph_26( + const uint32_t* vcg, // voxel connectivity graph + const int64_t sx, const int64_t sy, const int64_t sz, + OUT* out_labels = NULL, + size_t &N = _dummy_N +) { + const int64_t sxy = sx * sy; + const int64_t voxels = sx * sy * sz; + + uint64_t max_labels = static_cast(voxels) + 1; // + 1L for an array with no zeros + max_labels = std::min(max_labels, static_cast(std::numeric_limits::max())); + max_labels = std::min(max_labels, estimate_vcg_provisional_label_count(vcg, sx, voxels) + 1); + + if (out_labels == NULL) { + out_labels = new OUT[voxels](); + } + + DisjointSet equivalences(max_labels); + + /* + Layout of forward pass mask (which faces backwards). + N is the current location. + + z = -1 z = 0 + A B C J K L y = -1 + D E F M N y = 0 + G H I y = +1 + -1 0 +1 -1 0 <-- x axis + */ + + // Z - 1 + const int64_t A = -1 - sx - sxy; + const int64_t A_mask = 0b10000000000000000000000000; + const int64_t B = -sx - sxy; + const int64_t B_mask = 0b100000000000000000; + const int64_t C = +1 - sx - sxy; + const int64_t C_mask = 0b1000000000000000000000000; + const int64_t D = -1 - sxy; + const int64_t D_mask = 0b1000000000000000; + const int64_t E = -sxy; + const int64_t E_mask = 0b100000; + const int64_t F = +1 - sxy; + const int64_t F_mask = 0b100000000000000; + const int64_t G = -1 + sx - sxy; + const int64_t G_mask = 0b100000000000000000000000; + const int64_t H = +sx - sxy; + const int64_t H_mask = 0b10000000000000000; + const int64_t I = +1 + sx - sxy; + const int64_t I_mask = 0b10000000000000000000000; + + // Current Z + const int64_t J = -1 - sx; + const int64_t J_mask = 0b1000000000; + const int64_t K = -sx; + const int64_t K_mask = 0b1000; + const int64_t L = +1 - sx; + const int64_t L_mask = 0b100000000; + const int64_t M = -1; + const int64_t M_mask = 0b10; + // N = 0; + + OUT new_label = 0; + + for (int64_t z = 0; z < sz; z++) { + new_label++; + equivalences.add(new_label); + + for (int64_t x = 0; x < sx; x++) { + if (x > 0 && (vcg[x + sxy * z] & 0b0010) == 0) { + new_label++; + equivalences.add(new_label); + } + out_labels[x + sxy * z] = new_label; + } + + for (int64_t y = 1; y < sy; y++) { + for (int64_t x = 0; x < sx; x++) { + int64_t loc = x + sx * y + sxy * z; + + bool stolen = false; + + if (vcg[loc] & K_mask) { + out_labels[loc] = out_labels[loc+K]; + stolen = true; + } + if (x > 0 && (vcg[loc] & J_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+J]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+J]); + } + } + if (x > 0 && (vcg[loc] & M_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+M]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+M]); + } + } + if (x < sx - 1 && (vcg[loc] & L_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+L]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+L]); + } + } + + if (!stolen) { + new_label++; + out_labels[loc] = new_label; + equivalences.add(out_labels[loc]); + } + } + } + } + + for (int64_t z = 1; z < sz; z++) { + for (int64_t y = 0; y < sy; y++) { + for (int64_t x = 0; x < sx; x++) { + int64_t loc = x + sx * y + sxy * z; + + if (x > 0 && y > 0 && (vcg[loc] & A_mask)) { + equivalences.unify(out_labels[loc], out_labels[loc+A]); + } + if (y > 0 && (vcg[loc] & B_mask)) { + equivalences.unify(out_labels[loc], out_labels[loc+B]); + } + if (x < sx - 1 && y > 0 && (vcg[loc] & C_mask)) { + equivalences.unify(out_labels[loc], out_labels[loc+C]); + } + if (x > 0 && (vcg[loc] & D_mask)) { + equivalences.unify(out_labels[loc], out_labels[loc+D]); + } + if (vcg[loc] & E_mask) { + equivalences.unify(out_labels[loc], out_labels[loc+E]); + } + if (x < sx - 1 && (vcg[loc] & F_mask)) { + equivalences.unify(out_labels[loc], out_labels[loc+F]); + } + if (x > 0 && y < sy - 1 && (vcg[loc] & G_mask)) { + equivalences.unify(out_labels[loc], out_labels[loc+G]); + } + if (y < sy - 1 && (vcg[loc] & H_mask)) { + equivalences.unify(out_labels[loc], out_labels[loc+H]); + } + if (x < sx - 1 && y < sy - 1 && (vcg[loc] & I_mask)) { + equivalences.unify(out_labels[loc], out_labels[loc+I]); + } + } + } + } + + return simplified_relabel(out_labels, voxels, new_label, equivalences, N); +} + +template +OUT* color_connectivity_graph_6( + const uint8_t* vcg, // voxel connectivity graph + const int64_t sx, const int64_t sy, const int64_t sz, + OUT* out_labels = NULL, + size_t &N = _dummy_N +) { + throw new std::runtime_error("6-connectivity requires a 32-bit voxel graph for color_connectivity_graph_6."); +} + +template +OUT* color_connectivity_graph_6( + const uint32_t* vcg, // voxel connectivity graph + const int64_t sx, const int64_t sy, const int64_t sz, + OUT* out_labels = NULL, + size_t &N = _dummy_N +) { + const int64_t sxy = sx * sy; + const int64_t voxels = sx * sy * sz; + + uint64_t max_labels = static_cast(voxels) + 1; // + 1L for an array with no zeros + max_labels = std::min(max_labels, static_cast(std::numeric_limits::max())); + max_labels = std::min(max_labels, estimate_vcg_provisional_label_count(vcg, sx, voxels) + 1); + + if (out_labels == NULL) { + out_labels = new OUT[voxels](); + } + + DisjointSet equivalences(max_labels); + + const int64_t B = -1; + const int64_t B_mask = 0b10; + const int64_t C = -sx; + const int64_t C_mask = 0b1000; + + + OUT new_label = 0; + for (int64_t z = 0; z < sz; z++) { + new_label++; + equivalences.add(new_label); + + for (int64_t x = 0; x < sx; x++) { + if (x > 0 && (vcg[x + sxy * z] & B_mask) == 0) { + new_label++; + equivalences.add(new_label); + } + out_labels[x + sxy * z] = new_label; + } + + for (int64_t y = 1; y < sy; y++) { + for (int64_t x = 0; x < sx; x++) { + int64_t loc = x + sx * y + sxy * z; + + bool stolen = false; + + if (vcg[loc] & C_mask) { + out_labels[loc] = out_labels[loc+C]; + stolen = true; + } + if (x > 0 && (vcg[loc] & B_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+B]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+B]); + } + } + + if (!stolen) { + new_label++; + out_labels[loc] = new_label; + equivalences.add(out_labels[loc]); + } + } + } + } + + for (int64_t z = 1; z < sz; z++) { + for (int64_t y = 0; y < sy; y++) { + for (int64_t x = 0; x < sx; x++) { + int64_t loc = x + sx * y + sxy * z; + + if (vcg[loc] & 0b100000) { + equivalences.unify(out_labels[loc], out_labels[loc-sxy]); + } + } + } + } + + return simplified_relabel(out_labels, voxels, new_label, equivalences, N); +} + +// 8 and 32 bit inputs have different encodings so +// need to write this two ways +template +OUT* color_connectivity_graph_8( + const uint8_t* vcg, // voxel connectivity graph + const int64_t sx, const int64_t sy, + OUT* out_labels = NULL, + size_t &N = _dummy_N +) { + const int64_t sxy = sx * sy; + + uint64_t max_labels = static_cast(sxy) + 1; // + 1L for an array with no zeros + max_labels = std::min(max_labels, static_cast(std::numeric_limits::max())); + + if (out_labels == NULL) { + out_labels = new OUT[sxy](); + } + + DisjointSet equivalences(max_labels); + + OUT new_label = 1; + equivalences.add(new_label); + + for (int64_t x = 0; x < sx; x++) { + if (x > 0 && (vcg[x] & 0b0010) == 0) { + new_label++; + equivalences.add(new_label); + } + out_labels[x] = new_label; + } + + /* + Layout of mask. We start from e. + a | b | c + d | e | + */ + + const int64_t A = -1 - sx; + const int64_t A_mask = 0b10000000; + const int64_t B = -sx; + const int64_t B_mask = 0b1000; + const int64_t C = +1 - sx; + const int64_t C_mask = 0b1000000; + const int64_t D = -1; + const int64_t D_mask = 0b10; + + for (int64_t y = 1; y < sy; y++) { + for (int64_t x = 0; x < sx; x++) { + int64_t loc = x + sx * y; + + bool stolen = false; + + if (vcg[loc] & B_mask) { + out_labels[loc] = out_labels[loc+B]; + stolen = true; + } + if (x > 0 && (vcg[loc] & A_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+A]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+A]); + } + } + if (x > 0 && (vcg[loc] & D_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+D]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+D]); + } + } + if (x < sx - 1 && (vcg[loc] & C_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+C]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+C]); + } + } + + if (!stolen) { + new_label++; + out_labels[loc] = new_label; + equivalences.add(out_labels[loc]); + } + } + } + + return simplified_relabel(out_labels, sxy, new_label, equivalences, N); +} + +// 8 and 32 bit inputs have different encodings so +// need to write this two ways +template +OUT* color_connectivity_graph_8( + const uint32_t* vcg, // voxel connectivity graph + const int64_t sx, const int64_t sy, + OUT* out_labels = NULL, + size_t &N = _dummy_N +) { + const int64_t sxy = sx * sy; + + uint64_t max_labels = static_cast(sxy) + 1; // + 1L for an array with no zeros + max_labels = std::min(max_labels, static_cast(std::numeric_limits::max())); + + if (out_labels == NULL) { + out_labels = new OUT[sxy](); + } + + DisjointSet equivalences(max_labels); + + OUT new_label = 1; + equivalences.add(new_label); + + for (int64_t x = 0; x < sx; x++) { + if (x > 0 && (vcg[x] & 0b0010) == 0) { + new_label++; + equivalences.add(new_label); + } + out_labels[x] = new_label; + } + + /* + Layout of mask. We start from e. + a | b | c + d | e | + */ + + const int64_t A = -1 - sx; + const int64_t A_mask = 0b1000000000; + const int64_t B = -sx; + const int64_t B_mask = 0b1000; + const int64_t C = +1 - sx; + const int64_t C_mask = 0b100000000; + const int64_t D = -1; + const int64_t D_mask = 0b10; + + for (int64_t y = 1; y < sy; y++) { + for (int64_t x = 0; x < sx; x++) { + int64_t loc = x + sx * y; + + bool stolen = false; + + if (vcg[loc] & B_mask) { + out_labels[loc] = out_labels[loc+B]; + stolen = true; + } + if (x > 0 && (vcg[loc] & A_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+A]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+A]); + } + } + if (x > 0 && (vcg[loc] & D_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+D]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+D]); + } + } + if (x < sx - 1 && (vcg[loc] & C_mask)) { + if (!stolen) { + out_labels[loc] = out_labels[loc+C]; + stolen = true; + } + else { + equivalences.unify(out_labels[loc], out_labels[loc+C]); + } + } + + if (!stolen) { + new_label++; + out_labels[loc] = new_label; + equivalences.add(out_labels[loc]); + } + } + } + + return simplified_relabel(out_labels, sxy, new_label, equivalences, N); +} + +template +OUT* color_connectivity_graph_4( + const VCG_t* vcg, // voxel connectivity graph + const int64_t sx, const int64_t sy, + OUT* out_labels = NULL, + size_t &N = _dummy_N +) { + const int64_t sxy = sx * sy; + + uint64_t max_labels = static_cast(sxy) + 1; // + 1L for an array with no zeros + max_labels = std::min(max_labels, static_cast(std::numeric_limits::max())); + + if (out_labels == NULL) { + out_labels = new OUT[sxy](); + } + + DisjointSet equivalences(max_labels); + + OUT new_label = 1; + equivalences.add(new_label); + + for (int64_t x = 0; x < sx; x++) { + if (x > 0 && (vcg[x] & 0b0010) == 0) { + new_label++; + equivalences.add(new_label); + } + out_labels[x] = new_label; + } + + const int64_t B = -1; + const int64_t B_mask = 0b0010; + const int64_t C = -sx; + const int64_t C_mask = 0b1000; + + for (int64_t y = 1; y < sy; y++) { + for (int64_t x = 0; x < sx; x++) { + int64_t loc = x + sx * y; + + if (x > 0 && (vcg[loc] & B_mask)) { + out_labels[loc] = out_labels[loc+B]; + if (vcg[loc] & C_mask) { + equivalences.unify(out_labels[loc], out_labels[loc+C]); + } + } + else if (vcg[loc] & C_mask) { + out_labels[loc] = out_labels[loc+C]; + } + else { + new_label++; + out_labels[loc] = new_label; + equivalences.add(new_label); + } + } + } + + return simplified_relabel(out_labels, sxy, new_label, equivalences, N); +} + +template +OUT* color_connectivity_graph_N( + const VCG_t* vcg, // voxel connectivity graph + const int64_t sx, const int64_t sy, const int64_t sz, + const int connectivity, + OUT* out_labels = NULL, + size_t &N = _dummy_N +) { + if (connectivity != 26 && connectivity != 6 && connectivity != 4 && connectivity != 8) { + throw std::runtime_error("Only 4, 8, 6 and 26 connectivities are supported."); + } + + if (sz > 1 && (connectivity != 6 && connectivity != 26)) { + throw std::runtime_error("Only 6 and 26 connectivity is supported in 3D."); + } + + if (sz == 1) { + if (connectivity == 6 || connectivity == 4) { + return color_connectivity_graph_4(vcg, sx, sy, out_labels, N); + } + else { + return color_connectivity_graph_8(vcg, sx, sy, out_labels, N); + } + } + else if (connectivity == 6) { + return color_connectivity_graph_6(vcg, sx, sy, sz, out_labels, N); + } + else { + return color_connectivity_graph_26(vcg, sx, sy, sz, out_labels, N); + } +} + }; #endif \ No newline at end of file