mirror of
				https://github.com/omniscale/imposm3.git
				synced 2025-10-25 00:08:10 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			89 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			89 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package geos
 | |
| 
 | |
| /*
 | |
| #cgo LDFLAGS: -lgeos_c
 | |
| #include "geos_c.h"
 | |
| #include <stdint.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdarg.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| 
 | |
| extern void goLogString(char *msg);
 | |
| 
 | |
| void debug_wrap(const char *fmt, ...) {
 | |
| 	va_list a_list;
 | |
|     va_start(a_list, fmt);
 | |
| 
 | |
| 	char buf[1024];
 | |
| 	vsnprintf(buf, sizeof(buf), fmt, a_list);
 | |
| 	va_end(a_list);
 | |
| 	goLogString((char *)&buf);
 | |
| }
 | |
| 
 | |
| void devnull(const char *fmt, ...) {
 | |
| }
 | |
| 
 | |
| GEOSContextHandle_t initGEOS_r_debug() {
 | |
| 	return initGEOS_r(devnull, debug_wrap);
 | |
| }
 | |
| 
 | |
| void initGEOS_debug() {
 | |
|     return initGEOS(devnull, debug_wrap);
 | |
| }
 | |
| 
 | |
| typedef struct {
 | |
|     uint32_t num;
 | |
|     uint32_t *arr;
 | |
|     uint32_t arrCap;
 | |
| } queryResult;
 | |
| 
 | |
| void queryResultAppend(queryResult *r, int idx) {
 | |
|     if ((r->num+1) >= r->arrCap) {
 | |
|         uint32_t newCap = r->arrCap > 0 ? r->arrCap * 2 : 2;
 | |
|         uint32_t *newArr = malloc(sizeof(uint32_t) * newCap);
 | |
|         if (r->arrCap == 0) {
 | |
|             r->arr = newArr;
 | |
|         } else {
 | |
|             memcpy(newArr, r->arr, sizeof(uint32_t) * r->arrCap);
 | |
|             free(r->arr);
 | |
|             r->arr = newArr;
 | |
|         }
 | |
|         r->arrCap = newCap;
 | |
|     }
 | |
|     r->arr[r->num] = idx;
 | |
|     r->num += 1;
 | |
| }
 | |
| 
 | |
| void IndexQueryCallback(void *item, void *userdata) {
 | |
|     int idx = (size_t)item;
 | |
|     queryResult *result = (queryResult *)userdata;
 | |
|     queryResultAppend(result, idx);
 | |
| }
 | |
| 
 | |
| void IndexAdd(
 | |
|     GEOSContextHandle_t handle,
 | |
|     GEOSSTRtree *tree,
 | |
|     const GEOSGeometry *g,
 | |
|     size_t id)
 | |
| {
 | |
|     // instead of storing a void *, we just store our id
 | |
|     // this is safe since GEOS doesn't access the item pointer
 | |
|     GEOSSTRtree_insert_r(handle, tree, g, (void *)id);
 | |
| }
 | |
| 
 | |
| 
 | |
| // query with our custom callback
 | |
| uint32_t *IndexQuery(
 | |
|     GEOSContextHandle_t handle,
 | |
|     GEOSSTRtree *tree,
 | |
|     const GEOSGeometry *g,
 | |
|     uint32_t *num)
 | |
| {
 | |
|     queryResult result = {0};
 | |
|     GEOSSTRtree_query_r(handle, tree, g, IndexQueryCallback, &result);
 | |
|     *num = result.num;
 | |
|     return result.arr;
 | |
| }
 | |
| */
 | |
| import "C"
 | 
