parent
4c519106a7
commit
1c5a92440d
3 changed files with 657 additions and 2 deletions
@ -0,0 +1,653 @@ |
||||
// stb_connected_components - v0.90 - public domain connected components on grids
|
||||
// http://github.com/nothings/stb
|
||||
//
|
||||
// Finds connected components on 2D grids for testing reachability between
|
||||
// two points, with fast updates when changing reachability (e.g. on one machine
|
||||
// it was typically 0.2ms w/ 1024x1024 grid). Each grid square must be "open" or
|
||||
// "closed" (traversable or untraversable), and grid squares are only connected
|
||||
// to their orthogonal neighbors, not diagonally.
|
||||
//
|
||||
// In one source file, create the implementation by doing something like this:
|
||||
//
|
||||
// #define STBCC_GRID_COUNT_X_LOG2 10
|
||||
// #define STBCC_GRID_COUNT_Y_LOG2 10
|
||||
// #define STB_CONNECTED_COMPONENTS_IMPLEMENTATION
|
||||
// #include "stb_connected_components.h"
|
||||
//
|
||||
// The above creates an implementation that can run on maps up to 1024x1024.
|
||||
// Map sizes must be a multiple of 32 on each axis.
|
||||
//
|
||||
// LICENSE
|
||||
//
|
||||
// This software is dual-licensed to the public domain and under the following
|
||||
// license: you are granted a perpetual, irrevocable license to copy, modify,
|
||||
// publish, and distribute this file as you see fit.
|
||||
//
|
||||
|
||||
#ifndef INCLUDE_STB_CONNECTED_COMPONENTS_H |
||||
#define INCLUDE_STB_CONNECTED_COMPONENTS_H |
||||
|
||||
#include <stdlib.h> |
||||
#include <assert.h> |
||||
|
||||
typedef struct st_stbcc_grid stbcc_grid; |
||||
|
||||
//////////////////////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// initialization
|
||||
//
|
||||
|
||||
// you allocate the grid data structure to this size (note that it will be very big!!!)
|
||||
extern size_t stbcc_grid_sizeof(void); |
||||
|
||||
// initialize the grid, value of map[] is 0 = traversable, non-0 is solid
|
||||
extern void stbcc_init_grid(stbcc_grid *g, unsigned char *map, int w, int h); |
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// main functionality
|
||||
//
|
||||
|
||||
// update a grid square state, 0 = traversable, non-0 is solid
|
||||
// i can add a batch-update if it's needed
|
||||
extern void stbcc_update_grid(stbcc_grid *g, int x, int y, int solid); |
||||
|
||||
// query if two grid squares are reachable from each other
|
||||
extern int stbcc_query_grid_node_connection(stbcc_grid *g, int x1, int y1, int x2, int y2); |
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// bonus functions
|
||||
//
|
||||
|
||||
// query the grid data structure for whether a given square is open or not
|
||||
extern int stbcc_query_grid_open(stbcc_grid *g, int x, int y); |
||||
|
||||
// get a unique id for the connected component this is in; it's not necessarily
|
||||
// small, you'll need a hash table or something to remap it (or just use
|
||||
extern unsigned int stbcc_get_unique_id(stbcc_grid *g, int x, int y); |
||||
|
||||
#endif // INCLUDE_STB_CONNECTED_COMPONENTS_H
|
||||
|
||||
#ifdef STB_CONNECTED_COMPONENTS_IMPLEMENTATION |
||||
|
||||
#if !defined(STBCC_GRID_COUNT_X_LOG2) || !defined(STBCC_GRID_COUNT_Y_LOG2) |
||||
#error "You must define STBCC_GRID_COUNT_X_LOG2 and STBCC_GRID_COUNT_Y_LOG2 to define the max grid supported." |
||||
#endif |
||||
|
||||
#define STBCC__GRID_COUNT_X (1 << STBCC_GRID_COUNT_X_LOG2) |
||||
#define STBCC__GRID_COUNT_Y (1 << STBCC_GRID_COUNT_Y_LOG2) |
||||
|
||||
#define STBCC__MAP_STRIDE (1 << (STBCC_GRID_COUNT_X_LOG2-3)) |
||||
|
||||
#ifndef STBCC_CLUSTER_SIZE_X_LOG2 |
||||
#define STBCC_CLUSTER_SIZE_X_LOG2 5 |
||||
#endif |
||||
|
||||
#ifndef STBCC_CLUSTER_SIZE_Y_LOG2 |
||||
#define STBCC_CLUSTER_SIZE_Y_LOG2 5 |
||||
#endif |
||||
|
||||
#define STBCC__CLUSTER_SIZE_X (1 << STBCC_CLUSTER_SIZE_X_LOG2) |
||||
#define STBCC__CLUSTER_SIZE_Y (1 << STBCC_CLUSTER_SIZE_Y_LOG2) |
||||
|
||||
#define STBCC__CLUSTER_COUNT_X_LOG2 (STBCC_GRID_COUNT_X_LOG2 - STBCC_CLUSTER_SIZE_X_LOG2) |
||||
#define STBCC__CLUSTER_COUNT_Y_LOG2 (STBCC_GRID_COUNT_Y_LOG2 - STBCC_CLUSTER_SIZE_Y_LOG2) |
||||
|
||||
#define STBCC__CLUSTER_COUNT_X (1 << STBCC__CLUSTER_COUNT_X_LOG2) |
||||
#define STBCC__CLUSTER_COUNT_Y (1 << STBCC__CLUSTER_COUNT_Y_LOG2) |
||||
|
||||
#if STBCC__CLUSTER_SIZE_X >= STBCC__GRID_COUNT_X || STBCC__CLUSTER_SIZE_Y >= STBCC__GRID_COUNT_Y |
||||
#error "STBCC_CLUSTER_SIZE_X/Y_LOG2 must be smaller than STBCC_GRID_COUNT_X/Y_LOG2" |
||||
#endif |
||||
|
||||
// worst case # of clumps per cluster
|
||||
#define STBCC__MAX_CLUMPS_PER_CLUSTER_LOG2 (STBCC_CLUSTER_SIZE_X_LOG2 + STBCC_CLUSTER_SIZE_Y_LOG2-1) |
||||
#define STBCC__MAX_CLUMPS_PER_CLUSTER (1 << STBCC__MAX_CLUMPS_PER_CLUSTER_LOG2) |
||||
#define STBCC__MAX_CLUMPS (STBCC__MAX_CLUMPS_PER_CLUSTER * STBCC__CLUSTER_COUNT_X * STBCC__CLUSTER_COUNT_Y) |
||||
#define STBCC__NULL_CLUMPID STBCC__MAX_CLUMPS_PER_CLUSTER |
||||
|
||||
#define STBCC__CLUSTER_X_FOR_COORD_X(x) ((x) >> STBCC_CLUSTER_SIZE_X_LOG2) |
||||
#define STBCC__CLUSTER_Y_FOR_COORD_Y(y) ((y) >> STBCC_CLUSTER_SIZE_Y_LOG2) |
||||
|
||||
#define STBCC__MAP_BYTE_MASK(x,y) (1 << ((x) & 7)) |
||||
#define STBCC__MAP_BYTE(g,x,y) ((g)->map[y][(x) >> 3]) |
||||
#define STBCC__MAP_OPEN(g,x,y) (STBCC__MAP_BYTE(g,x,y) & STBCC__MAP_BYTE_MASK(x,y)) |
||||
|
||||
typedef unsigned short stbcc__clumpid; |
||||
typedef unsigned char stbcc__verify_max_clumps[STBCC__MAX_CLUMPS_PER_CLUSTER < (1 << (8*sizeof(stbcc__clumpid))) ? 1 : -1]; |
||||
|
||||
#define STBCC__MAX_EXITS_PER_CLUMP (STBCC__CLUSTER_SIZE_X + STBCC__CLUSTER_SIZE_Y) // 64
|
||||
// 2^19 * 2^6 => 2^25 exits => 2^26 => 64MB for 1024x1024
|
||||
|
||||
typedef unsigned char stbcc__verify_max_exits[STBCC__MAX_EXITS_PER_CLUMP <= 256]; |
||||
|
||||
typedef struct |
||||
{ |
||||
unsigned short clump_index:12; |
||||
signed short cluster_dx:2; |
||||
signed short cluster_dy:2; |
||||
} stbcc__relative_clumpid; |
||||
|
||||
typedef union |
||||
{ |
||||
struct { |
||||
unsigned int clump_index:12; |
||||
unsigned int cluster_x:10; |
||||
unsigned int cluster_y:10; |
||||
} f; |
||||
unsigned int c; |
||||
} stbcc__global_clumpid; |
||||
|
||||
// rebuilt cluster 3,4
|
||||
|
||||
// what changes in cluster 2,4
|
||||
|
||||
typedef struct |
||||
{ |
||||
stbcc__global_clumpid global_label; |
||||
unsigned char num_adjacent; |
||||
stbcc__relative_clumpid adjacent_clumps[STBCC__MAX_EXITS_PER_CLUMP]; |
||||
} stbcc__clump; |
||||
|
||||
typedef struct |
||||
{ |
||||
unsigned int num_clumps; |
||||
stbcc__clump clump[STBCC__MAX_CLUMPS_PER_CLUSTER]; |
||||
} stbcc__cluster; |
||||
|
||||
struct st_stbcc_grid |
||||
{ |
||||
int w,h,cw,ch; |
||||
unsigned char map[STBCC__GRID_COUNT_Y][STBCC__MAP_STRIDE]; // 1K x 1K => 1K x 128 => 128KB
|
||||
stbcc__clumpid clump_for_node[STBCC__GRID_COUNT_Y][STBCC__GRID_COUNT_X]; // 1K x 1K x 2 = 2MB
|
||||
stbcc__cluster cluster[STBCC__CLUSTER_COUNT_Y][STBCC__CLUSTER_COUNT_X]; // 1K x 1K x 0.5 x 64 x 2 = 64MB
|
||||
}; |
||||
|
||||
int stbcc_query_grid_node_connection(stbcc_grid *g, int x1, int y1, int x2, int y2) |
||||
{ |
||||
stbcc__global_clumpid label1, label2; |
||||
stbcc__clumpid c1 = g->clump_for_node[y1][x1]; |
||||
stbcc__clumpid c2 = g->clump_for_node[y2][x2]; |
||||
int cx1 = STBCC__CLUSTER_X_FOR_COORD_X(x1); |
||||
int cy1 = STBCC__CLUSTER_Y_FOR_COORD_Y(y1); |
||||
int cx2 = STBCC__CLUSTER_X_FOR_COORD_X(x2); |
||||
int cy2 = STBCC__CLUSTER_Y_FOR_COORD_Y(y2); |
||||
if (c1 == STBCC__NULL_CLUMPID || c2 == STBCC__NULL_CLUMPID) |
||||
return 0; |
||||
label1 = g->cluster[cy1][cx1].clump[c1].global_label; |
||||
label2 = g->cluster[cy2][cx2].clump[c2].global_label; |
||||
if (label1.c == label2.c) |
||||
return 1; |
||||
return 0; |
||||
} |
||||
|
||||
int stbcc_query_grid_open(stbcc_grid *g, int x, int y) |
||||
{ |
||||
return STBCC__MAP_OPEN(g, x, y) != 0; |
||||
} |
||||
|
||||
unsigned int stbcc_get_unique_id(stbcc_grid *g, int x, int y) |
||||
{ |
||||
stbcc__clumpid c = g->clump_for_node[y][x]; |
||||
int cx = STBCC__CLUSTER_X_FOR_COORD_X(x); |
||||
int cy = STBCC__CLUSTER_Y_FOR_COORD_Y(y); |
||||
return g->cluster[cy][cx].clump[c].global_label.c; |
||||
} |
||||
|
||||
typedef struct |
||||
{ |
||||
unsigned char x,y; |
||||
} stbcc__tinypoint; |
||||
|
||||
typedef struct |
||||
{ |
||||
stbcc__tinypoint parent[STBCC__CLUSTER_SIZE_Y][STBCC__CLUSTER_SIZE_X]; // 32x32 => 2KB
|
||||
stbcc__clumpid label[STBCC__CLUSTER_SIZE_Y][STBCC__CLUSTER_SIZE_X]; |
||||
} stbcc__cluster_build_info; |
||||
|
||||
static void stbcc__build_clumps_for_cluster(stbcc_grid *g, int cx, int cy); |
||||
static void stbcc__remove_connections_to_adjacent_cluster(stbcc_grid *g, int cx, int cy, int dx, int dy); |
||||
static void stbcc__add_connections_to_adjacent_cluster(stbcc_grid *g, int cx, int cy, int dx, int dy); |
||||
|
||||
static stbcc__global_clumpid stbcc__clump_find(stbcc_grid *g, stbcc__global_clumpid n) |
||||
{ |
||||
stbcc__global_clumpid q; |
||||
stbcc__clump *c = &g->cluster[n.f.cluster_y][n.f.cluster_x].clump[n.f.clump_index]; |
||||
|
||||
if (c->global_label.c == n.c) |
||||
return n; |
||||
|
||||
q = stbcc__clump_find(g, c->global_label); |
||||
c->global_label = q; |
||||
return q; |
||||
} |
||||
|
||||
typedef struct |
||||
{ |
||||
unsigned int cluster_x; |
||||
unsigned int cluster_y; |
||||
unsigned int clump_index; |
||||
} stbcc__unpacked_clumpid; |
||||
|
||||
// @OPTIMIZE: pass in these parameters unpacked, not packed
|
||||
static void stbcc__clump_union(stbcc_grid *g, stbcc__unpacked_clumpid m, int x, int y, int idx) |
||||
{ |
||||
stbcc__clump *mc = &g->cluster[m.cluster_y][m.cluster_x].clump[m.clump_index]; |
||||
stbcc__clump *nc = &g->cluster[y][x].clump[idx]; |
||||
stbcc__global_clumpid mp = stbcc__clump_find(g, mc->global_label); |
||||
stbcc__global_clumpid np = stbcc__clump_find(g, nc->global_label); |
||||
|
||||
if (mp.c == np.c) |
||||
return; |
||||
|
||||
g->cluster[mp.f.cluster_y][mp.f.cluster_x].clump[mp.f.clump_index].global_label = np; |
||||
} |
||||
|
||||
static void stbcc__build_connected_components_for_clumps(stbcc_grid *g) |
||||
{ |
||||
int i,j,k,h; |
||||
|
||||
for (j=0; j < STBCC__CLUSTER_COUNT_Y; ++j) { |
||||
for (i=0; i < STBCC__CLUSTER_COUNT_X; ++i) { |
||||
stbcc__cluster *cluster = &g->cluster[j][i]; |
||||
for (k=0; k < (int) cluster->num_clumps; ++k) { |
||||
stbcc__global_clumpid m; |
||||
m.f.clump_index = k; |
||||
m.f.cluster_x = i; |
||||
m.f.cluster_y = j; |
||||
assert((int) m.f.clump_index == k && (int) m.f.cluster_x == i && (int) m.f.cluster_y == j); |
||||
cluster->clump[k].global_label = m; |
||||
} |
||||
} |
||||
} |
||||
|
||||
for (j=0; j < STBCC__CLUSTER_COUNT_Y; ++j) { |
||||
for (i=0; i < STBCC__CLUSTER_COUNT_X; ++i) { |
||||
stbcc__cluster *cluster = &g->cluster[j][i]; |
||||
for (k=0; k < (int) cluster->num_clumps; ++k) { |
||||
stbcc__clump *clump = &cluster->clump[k]; |
||||
stbcc__unpacked_clumpid m; |
||||
m.clump_index = k; |
||||
m.cluster_x = i; |
||||
m.cluster_y = j; |
||||
for (h=0; h < clump->num_adjacent; ++h) { |
||||
unsigned int clump_index = clump->adjacent_clumps[h].clump_index; |
||||
unsigned int x = clump->adjacent_clumps[h].cluster_dx + i; |
||||
unsigned int y = clump->adjacent_clumps[h].cluster_dy + j; |
||||
stbcc__clump_union(g, m, x, y, clump_index); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
for (j=0; j < STBCC__CLUSTER_COUNT_Y; ++j) { |
||||
for (i=0; i < STBCC__CLUSTER_COUNT_X; ++i) { |
||||
stbcc__cluster *cluster = &g->cluster[j][i]; |
||||
for (k=0; k < (int) cluster->num_clumps; ++k) { |
||||
stbcc__global_clumpid m; |
||||
m.f.clump_index = k; |
||||
m.f.cluster_x = i; |
||||
m.f.cluster_y = j; |
||||
stbcc__clump_find(g, m); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
void stbcc_update_grid(stbcc_grid *g, int x, int y, int solid) |
||||
{ |
||||
int cx,cy; |
||||
|
||||
if (!solid) { |
||||
if (STBCC__MAP_OPEN(g,x,y)) |
||||
return; |
||||
} else { |
||||
if (!STBCC__MAP_OPEN(g,x,y)) |
||||
return; |
||||
} |
||||
|
||||
cx = STBCC__CLUSTER_X_FOR_COORD_X(x); |
||||
cy = STBCC__CLUSTER_Y_FOR_COORD_Y(y); |
||||
|
||||
stbcc__remove_connections_to_adjacent_cluster(g, cx-1, cy, 1, 0); |
||||
stbcc__remove_connections_to_adjacent_cluster(g, cx+1, cy, -1, 0); |
||||
stbcc__remove_connections_to_adjacent_cluster(g, cx, cy-1, 0, 1); |
||||
stbcc__remove_connections_to_adjacent_cluster(g, cx, cy+1, 0,-1); |
||||
|
||||
if (!solid) |
||||
STBCC__MAP_BYTE(g,x,y) |= STBCC__MAP_BYTE_MASK(x,y); |
||||
else |
||||
STBCC__MAP_BYTE(g,x,y) &= ~STBCC__MAP_BYTE_MASK(x,y); |
||||
|
||||
stbcc__build_clumps_for_cluster(g, cx, cy); |
||||
|
||||
stbcc__add_connections_to_adjacent_cluster(g, cx, cy, -1, 0); |
||||
stbcc__add_connections_to_adjacent_cluster(g, cx, cy, 1, 0); |
||||
stbcc__add_connections_to_adjacent_cluster(g, cx, cy, 0,-1); |
||||
stbcc__add_connections_to_adjacent_cluster(g, cx, cy, 0, 1); |
||||
|
||||
stbcc__add_connections_to_adjacent_cluster(g, cx-1, cy, 1, 0); |
||||
stbcc__add_connections_to_adjacent_cluster(g, cx+1, cy, -1, 0); |
||||
stbcc__add_connections_to_adjacent_cluster(g, cx, cy-1, 0, 1); |
||||
stbcc__add_connections_to_adjacent_cluster(g, cx, cy+1, 0,-1); |
||||
|
||||
stbcc__build_connected_components_for_clumps(g); |
||||
} |
||||
|
||||
size_t stbcc_grid_sizeof(void) |
||||
{ |
||||
return sizeof(stbcc_grid); |
||||
} |
||||
|
||||
void stbcc_init_grid(stbcc_grid *g, unsigned char *map, int w, int h) |
||||
{ |
||||
int i,j,k; |
||||
assert(w % STBCC__CLUSTER_SIZE_X == 0); |
||||
assert(h % STBCC__CLUSTER_SIZE_Y == 0); |
||||
assert(w % 8 == 0); |
||||
|
||||
g->w = w; |
||||
g->h = h; |
||||
g->cw = w >> STBCC_CLUSTER_SIZE_X_LOG2; |
||||
g->ch = h >> STBCC_CLUSTER_SIZE_Y_LOG2; |
||||
|
||||
for (j=0; j < h; ++j) { |
||||
for (i=0; i < w; i += 8) { |
||||
unsigned char c = 0; |
||||
for (k=0; k < 8; ++k) |
||||
if (map[j*w + (i+k)] == 0) |
||||
c |= (1 << k); |
||||
g->map[j][i>>3] = c; |
||||
} |
||||
} |
||||
|
||||
for (j=0; j < g->ch; ++j) |
||||
for (i=0; i < g->cw; ++i) |
||||
stbcc__build_clumps_for_cluster(g, i, j); |
||||
|
||||
for (j=0; j < g->ch; ++j) { |
||||
for (i=0; i < g->cw; ++i) { |
||||
stbcc__add_connections_to_adjacent_cluster(g, i, j, -1, 0); |
||||
stbcc__add_connections_to_adjacent_cluster(g, i, j, 1, 0); |
||||
stbcc__add_connections_to_adjacent_cluster(g, i, j, 0,-1); |
||||
stbcc__add_connections_to_adjacent_cluster(g, i, j, 0, 1); |
||||
} |
||||
} |
||||
|
||||
stbcc__build_connected_components_for_clumps(g); |
||||
|
||||
for (j=0; j < g->h; ++j) |
||||
for (i=0; i < g->w; ++i) |
||||
assert(g->clump_for_node[j][i] <= STBCC__NULL_CLUMPID); |
||||
} |
||||
|
||||
|
||||
static void stbcc__add_clump_connection(stbcc_grid *g, int x1, int y1, int x2, int y2) |
||||
{ |
||||
stbcc__clump *clump; |
||||
|
||||
int cx1 = STBCC__CLUSTER_X_FOR_COORD_X(x1); |
||||
int cy1 = STBCC__CLUSTER_Y_FOR_COORD_Y(y1); |
||||
int cx2 = STBCC__CLUSTER_X_FOR_COORD_X(x2); |
||||
int cy2 = STBCC__CLUSTER_Y_FOR_COORD_Y(y2); |
||||
|
||||
stbcc__clumpid c1 = g->clump_for_node[y1][x1]; |
||||
stbcc__clumpid c2 = g->clump_for_node[y2][x2]; |
||||
|
||||
stbcc__relative_clumpid rc; |
||||
|
||||
assert(cx1 != cx2 || cy1 != cy2); |
||||
assert(abs(cx1-cx2) + abs(cy1-cy2) == 1); |
||||
|
||||
// add connection to c2 in c1
|
||||
|
||||
rc.clump_index = c2; |
||||
rc.cluster_dx = x2-x1; |
||||
rc.cluster_dy = y2-y1; |
||||
|
||||
clump = &g->cluster[cy1][cx1].clump[c1]; |
||||
assert(clump->num_adjacent < STBCC__MAX_EXITS_PER_CLUMP); |
||||
clump->adjacent_clumps[clump->num_adjacent++] = rc; |
||||
} |
||||
|
||||
static void stbcc__remove_clump_connection(stbcc_grid *g, int x1, int y1, int x2, int y2) |
||||
{ |
||||
stbcc__clump *clump; |
||||
int i; |
||||
|
||||
int cx1 = STBCC__CLUSTER_X_FOR_COORD_X(x1); |
||||
int cy1 = STBCC__CLUSTER_Y_FOR_COORD_Y(y1); |
||||
int cx2 = STBCC__CLUSTER_X_FOR_COORD_X(x2); |
||||
int cy2 = STBCC__CLUSTER_Y_FOR_COORD_Y(y2); |
||||
|
||||
stbcc__clumpid c1 = g->clump_for_node[y1][x1]; |
||||
stbcc__clumpid c2 = g->clump_for_node[y2][x2]; |
||||
|
||||
stbcc__relative_clumpid rc; |
||||
|
||||
assert(cx1 != cx2 || cy1 != cy2); |
||||
assert(abs(cx1-cx2) + abs(cy1-cy2) == 1); |
||||
|
||||
// add connection to c2 in c1
|
||||
|
||||
rc.clump_index = c2; |
||||
rc.cluster_dx = x2-x1; |
||||
rc.cluster_dy = y2-y1; |
||||
|
||||
clump = &g->cluster[cy1][cx1].clump[c1]; |
||||
|
||||
for (i=0; i < clump->num_adjacent; ++i) |
||||
if (rc.clump_index == clump->adjacent_clumps[i].clump_index && |
||||
rc.cluster_dx == clump->adjacent_clumps[i].cluster_dx && |
||||
rc.cluster_dy == clump->adjacent_clumps[i].cluster_dy)
|
||||
break; |
||||
|
||||
if (i < clump->num_adjacent) |
||||
clump->adjacent_clumps[i] = clump->adjacent_clumps[--clump->num_adjacent]; |
||||
else |
||||
assert(0); |
||||
} |
||||
|
||||
static void stbcc__add_connections_to_adjacent_cluster(stbcc_grid *g, int cx, int cy, int dx, int dy) |
||||
{ |
||||
unsigned char connected[STBCC__MAX_CLUMPS_PER_CLUSTER/8] = { 0 }; |
||||
int x = cx * STBCC__CLUSTER_SIZE_X; |
||||
int y = cy * STBCC__CLUSTER_SIZE_Y; |
||||
int step_x, step_y=0, i, j, k, n; |
||||
|
||||
if (cx < 0 || cx >= g->cw || cy < 0 || cy >= g->ch) |
||||
return; |
||||
|
||||
if (cx+dx < 0 || cx+dx >= g->cw || cy+dy < 0 || cy+dy >= g->ch) |
||||
return; |
||||
|
||||
assert(abs(dx) + abs(dy) == 1); |
||||
|
||||
if (dx == 1) { |
||||
i = STBCC__CLUSTER_SIZE_X-1; |
||||
j = 0; |
||||
step_x = 0; |
||||
step_y = 1; |
||||
n = STBCC__CLUSTER_SIZE_Y; |
||||
} else if (dx == -1) { |
||||
i = 0; |
||||
j = 0; |
||||
step_x = 0; |
||||
step_y = 1;
|
||||
n = STBCC__CLUSTER_SIZE_Y; |
||||
} else if (dy == -1) { |
||||
i = 0; |
||||
j = 0; |
||||
step_x = 1; |
||||
step_y = 0; |
||||
n = STBCC__CLUSTER_SIZE_X; |
||||
} else if (dy == 1) { |
||||
i = 0; |
||||
j = STBCC__CLUSTER_SIZE_Y-1; |
||||
step_x = 1; |
||||
step_y = 0; |
||||
n = STBCC__CLUSTER_SIZE_X; |
||||
} else { |
||||
assert(0); |
||||
} |
||||
|
||||
for (k=0; k < n; ++k) { |
||||
if (STBCC__MAP_OPEN(g, x+i, y+j) && STBCC__MAP_OPEN(g, x+i+dx, y+j+dy)) { |
||||
stbcc__clumpid c = g->clump_for_node[y+j+dy][x+i+dx]; |
||||
if (0 == (connected[c>>3] & (1 << (c & 7)))) { |
||||
connected[c>>3] |= 1 << (c & 7); |
||||
stbcc__add_clump_connection(g, x+i, y+j, x+i+dx, y+j+dy); |
||||
} |
||||
} |
||||
i += step_x; |
||||
j += step_y; |
||||
} |
||||
} |
||||
|
||||
static void stbcc__remove_connections_to_adjacent_cluster(stbcc_grid *g, int cx, int cy, int dx, int dy) |
||||
{ |
||||
unsigned char disconnected[STBCC__MAX_CLUMPS_PER_CLUSTER/8] = { 0 }; |
||||
int x = cx * STBCC__CLUSTER_SIZE_X; |
||||
int y = cy * STBCC__CLUSTER_SIZE_Y; |
||||
int step_x, step_y=0, i, j, k, n; |
||||
|
||||
if (cx < 0 || cx >= g->cw || cy < 0 || cy >= g->ch) |
||||
return; |
||||
|
||||
if (cx+dx < 0 || cx+dx >= g->cw || cy+dy < 0 || cy+dy >= g->ch) |
||||
return; |
||||
|
||||
assert(abs(dx) + abs(dy) == 1); |
||||
|
||||
if (dx == 1) { |
||||
i = STBCC__CLUSTER_SIZE_X-1; |
||||
j = 0; |
||||
step_x = 0; |
||||
step_y = 1; |
||||
n = STBCC__CLUSTER_SIZE_Y; |
||||
} else if (dx == -1) { |
||||
i = 0; |
||||
j = 0; |
||||
step_x = 0; |
||||
step_y = 1;
|
||||
n = STBCC__CLUSTER_SIZE_Y; |
||||
} else if (dy == -1) { |
||||
i = 0; |
||||
j = 0; |
||||
step_x = 1; |
||||
step_y = 0; |
||||
n = STBCC__CLUSTER_SIZE_X; |
||||
} else if (dy == 1) { |
||||
i = 0; |
||||
j = STBCC__CLUSTER_SIZE_Y-1; |
||||
step_x = 1; |
||||
step_y = 0; |
||||
n = STBCC__CLUSTER_SIZE_X; |
||||
} else { |
||||
assert(0); |
||||
} |
||||
|
||||
for (k=0; k < n; ++k) { |
||||
if (STBCC__MAP_OPEN(g, x+i, y+j) && STBCC__MAP_OPEN(g, x+i+dx, y+j+dy)) { |
||||
stbcc__clumpid c = g->clump_for_node[y+j+dy][x+i+dx]; |
||||
if (0 == (disconnected[c>>3] & (1 << (c & 7)))) { |
||||
disconnected[c>>3] |= 1 << (c & 7); |
||||
stbcc__remove_clump_connection(g, x+i, y+j, x+i+dx, y+j+dy); |
||||
} |
||||
} |
||||
i += step_x; |
||||
j += step_y; |
||||
} |
||||
} |
||||
|
||||
static stbcc__tinypoint stbcc__incluster_find(stbcc__cluster_build_info *cbi, int x, int y) |
||||
{ |
||||
stbcc__tinypoint p,q; |
||||
p = cbi->parent[y][x]; |
||||
if (p.x == x && p.y == y) |
||||
return p; |
||||
q = stbcc__incluster_find(cbi, p.x, p.y); |
||||
cbi->parent[y][x] = q; |
||||
return q; |
||||
} |
||||
|
||||
static void stbcc__incluster_union(stbcc__cluster_build_info *cbi, int x1, int y1, int x2, int y2) |
||||
{ |
||||
stbcc__tinypoint p = stbcc__incluster_find(cbi, x1,y1); |
||||
stbcc__tinypoint q = stbcc__incluster_find(cbi, x2,y2); |
||||
|
||||
if (p.x == q.x && p.y == q.y) |
||||
return; |
||||
|
||||
cbi->parent[p.y][p.x] = q; |
||||
} |
||||
|
||||
static void stbcc__build_clumps_for_cluster(stbcc_grid *g, int cx, int cy) |
||||
{ |
||||
stbcc__cluster *c; |
||||
stbcc__cluster_build_info cbi; |
||||
int label=0; |
||||
int i,j; |
||||
int x = cx * STBCC__CLUSTER_SIZE_X; |
||||
int y = cy * STBCC__CLUSTER_SIZE_Y; |
||||
|
||||
// set initial disjoint set forest state
|
||||
for (j=0; j < STBCC__CLUSTER_SIZE_Y; ++j) { |
||||
for (i=0; i < STBCC__CLUSTER_SIZE_X; ++i) { |
||||
cbi.parent[j][i].x = i; |
||||
cbi.parent[j][i].y = j; |
||||
} |
||||
} |
||||
|
||||
// join all sets that are connected
|
||||
for (j=0; j < STBCC__CLUSTER_SIZE_Y; ++j) { |
||||
// check down only if not on bottom row
|
||||
if (j < STBCC__CLUSTER_SIZE_Y-1) |
||||
for (i=0; i < STBCC__CLUSTER_SIZE_X; ++i) |
||||
if (STBCC__MAP_OPEN(g,x+i,y+j) && STBCC__MAP_OPEN(g,x+i ,y+j+1)) |
||||
stbcc__incluster_union(&cbi, i,j, i,j+1); |
||||
// check right for everything but rightmost column
|
||||
for (i=0; i < STBCC__CLUSTER_SIZE_X-1; ++i) |
||||
if (STBCC__MAP_OPEN(g,x+i,y+j) && STBCC__MAP_OPEN(g,x+i+1,y+j )) |
||||
stbcc__incluster_union(&cbi, i,j, i+1,j); |
||||
} |
||||
|
||||
// label all non-empty leaders
|
||||
for (j=0; j < STBCC__CLUSTER_SIZE_Y; ++j) { |
||||
for (i=0; i < STBCC__CLUSTER_SIZE_X; ++i) { |
||||
stbcc__tinypoint p = cbi.parent[j][i]; |
||||
if (p.x == i && p.y == j) |
||||
if (STBCC__MAP_OPEN(g,x+i,y+j)) |
||||
cbi.label[j][i] = label++; |
||||
else |
||||
cbi.label[j][i] = STBCC__NULL_CLUMPID; |
||||
} |
||||
} |
||||
|
||||
// label all other nodes
|
||||
for (j=0; j < STBCC__CLUSTER_SIZE_Y; ++j) { |
||||
for (i=0; i < STBCC__CLUSTER_SIZE_X; ++i) { |
||||
stbcc__tinypoint p = stbcc__incluster_find(&cbi, i,j); |
||||
if (p.x != i || p.y != j) { |
||||
if (STBCC__MAP_OPEN(g,x+i,y+j)) |
||||
cbi.label[j][i] = cbi.label[p.y][p.x]; |
||||
} |
||||
} |
||||
} |
||||
|
||||
c = &g->cluster[cy][cx]; |
||||
c->num_clumps = label; |
||||
for (i=0; i < label; ++i) |
||||
c->clump[i].num_adjacent = 0; |
||||
|
||||
for (j=0; j < STBCC__CLUSTER_SIZE_Y; ++j) |
||||
for (i=0; i < STBCC__CLUSTER_SIZE_X; ++i) { |
||||
g->clump_for_node[y+j][x+i] = cbi.label[j][i]; // @OPTIMIZE: remove cbi.label entirely
|
||||
assert(g->clump_for_node[y+j][x+i] <= STBCC__NULL_CLUMPID); |
||||
} |
||||
} |
||||
|
||||
#endif // STB_CONNECTED_COMPONENTS_IMPLEMENTATION
|
Loading…
Reference in New Issue