[postgis-tickets] r14507 - #3404, ST_ClusterWithin crashes backend

Daniel Baston dbaston at gmail.com
Sun Dec 20 13:29:38 PST 2015


Author: dbaston
Date: 2015-12-20 13:29:38 -0800 (Sun, 20 Dec 2015)
New Revision: 14507

Modified:
   trunk/liblwgeom/cunit/cu_unionfind.c
   trunk/liblwgeom/lwgeom_geos_cluster.c
   trunk/liblwgeom/lwunionfind.c
Log:
#3404, ST_ClusterWithin crashes backend

Modified: trunk/liblwgeom/cunit/cu_unionfind.c
===================================================================
--- trunk/liblwgeom/cunit/cu_unionfind.c	2015-12-19 07:45:14 UTC (rev 14506)
+++ trunk/liblwgeom/cunit/cu_unionfind.c	2015-12-20 21:29:38 UTC (rev 14507)
@@ -87,6 +87,29 @@
 	lwfree(ids_by_cluster);
 }
 
+static void test_unionfind_path_compression(void)
+{
+	UNIONFIND* uf = UF_create(5);
+	uint32_t i;
+
+	uf->clusters[1] = 0;
+	uf->clusters[2] = 1;
+	uf->clusters[3] = 2;
+	uf->clusters[4] = 3;
+
+	/* Calling "find" on a leaf should attach all nodes between the root and the
+	 * leaf directly to the root. */
+	uint32_t root = UF_find(uf, 4);
+	for (i = 0; i < uf->N; i++)
+	{
+		/* Verify that all cluster references have been updated to point
+		 * directly to the root. */
+		CU_ASSERT_EQUAL(root, uf->clusters[i]);
+	}
+
+	UF_destroy(uf);
+}
+
 void unionfind_suite_setup(void);
 void unionfind_suite_setup(void)
 {
@@ -94,4 +117,5 @@
 	PG_ADD_TEST(suite, test_unionfind_create);
 	PG_ADD_TEST(suite, test_unionfind_union);
 	PG_ADD_TEST(suite, test_unionfind_ordered_by_cluster);
+	PG_ADD_TEST(suite, test_unionfind_path_compression);
 }

Modified: trunk/liblwgeom/lwgeom_geos_cluster.c
===================================================================
--- trunk/liblwgeom/lwgeom_geos_cluster.c	2015-12-19 07:45:14 UTC (rev 14506)
+++ trunk/liblwgeom/lwgeom_geos_cluster.c	2015-12-20 21:29:38 UTC (rev 14507)
@@ -349,10 +349,15 @@
 		if ((i == num_geoms - 1) ||
 		        (UF_find(uf, ordered_components[i]) != UF_find(uf, ordered_components[i+1])))
 		{
+			if (k >= uf->num_clusters) {
+				/* Should not get here - it means that we have more clusters than uf->num_clusters thinks we should. */
+				return LW_FAILURE;
+			}
+
 			if (is_lwgeom)
 			{
-				LWGEOM** components = lwalloc(num_geoms * sizeof(LWGEOM*));
-				memcpy(components, geoms_in_cluster, num_geoms * sizeof(LWGEOM*));
+				LWGEOM** components = lwalloc(j * sizeof(LWGEOM*));
+				memcpy(components, geoms_in_cluster, j * sizeof(LWGEOM*));
 				(*clusterGeoms)[k++] = lwcollection_construct(COLLECTIONTYPE, components[0]->srid, NULL, j, (LWGEOM**) components);
 			}
 			else

Modified: trunk/liblwgeom/lwunionfind.c
===================================================================
--- trunk/liblwgeom/lwunionfind.c	2015-12-19 07:45:14 UTC (rev 14506)
+++ trunk/liblwgeom/lwunionfind.c	2015-12-20 21:29:38 UTC (rev 14507)
@@ -60,11 +60,17 @@
 uint32_t
 UF_find (UNIONFIND* uf, uint32_t i)
 {
-	while (uf->clusters[i] != i)
-	{
-		uf->clusters[i] = uf->clusters[uf->clusters[i]];
-		i = uf->clusters[i];
+	uint32_t base = i;
+	while (uf->clusters[base] != base) {
+		base = uf->clusters[base];
 	}
+
+	while (i != base) {
+		uint32_t next = uf->clusters[i];
+		uf->clusters[i] = base;
+		i = next;
+	}
+
 	return i;
 }
 



More information about the postgis-tickets mailing list