| 
									
										
										
										
											2009-02-16 12:23:29 +00:00
										 |  |  | #if defined MYDDAS_MYSQL
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include "Yap.h"
 | 
					
						
							|  |  |  | #include <netinet/in.h>
 | 
					
						
							|  |  |  | #include "myddas_wkb.h"
 | 
					
						
							|  |  |  | #include "myddas_wkb2prolog.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void readswap4(uint32 *buf); | 
					
						
							|  |  |  | static void readswap8(double *buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static byte get_hostbyteorder(void); | 
					
						
							|  |  |  | static byte get_inbyteorder(void); | 
					
						
							|  |  |  | static uint32 get_wkbType(void); | 
					
						
							|  |  |  | static Term get_point(char *functor); | 
					
						
							|  |  |  | static Term get_linestring(char *functor); | 
					
						
							|  |  |  | static Term get_polygon(char *functor); | 
					
						
							|  |  |  | static Term get_geometry(uint32 type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int swaporder; | 
					
						
							|  |  |  | static byte inbyteorder, hostbyteorder; | 
					
						
							|  |  |  | static byte *cursor; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Term wkb2prolog(char *wkb) { | 
					
						
							|  |  |  |   uint32 type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   cursor = wkb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /*ignore the SRID 4 bytes*/ | 
					
						
							|  |  |  |   cursor += 4; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /*byteorder*/ | 
					
						
							|  |  |  |   hostbyteorder = get_hostbyteorder(); | 
					
						
							|  |  |  |   inbyteorder = get_inbyteorder(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   swaporder = 0; | 
					
						
							|  |  |  |   if ( hostbyteorder != inbyteorder ) | 
					
						
							|  |  |  |     swaporder = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   type = get_wkbType(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return get_geometry(type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static byte get_hostbyteorder(void){ | 
					
						
							|  |  |  |   uint16_t host = 5; | 
					
						
							|  |  |  |   uint16_t net; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   net = htons(host); | 
					
						
							|  |  |  |   if ( net == host ) | 
					
						
							|  |  |  |     return(WKBXDR); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     return(WKBNDR); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static byte get_inbyteorder(void){ | 
					
						
							|  |  |  |   byte b = cursor[0]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (b != WKBNDR && b != WKBXDR) { | 
					
						
							|  |  |  |     fprintf(stderr, "Unknown byteorder: %d\n",b); | 
					
						
							|  |  |  |     exit(0); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   cursor++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return(b); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint32 get_wkbType(void){ | 
					
						
							|  |  |  |   uint32 u; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* read the type */ | 
					
						
							|  |  |  |   readswap4(&u); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (u > WKBMAXTYPE || u < WKBMINTYPE) { | 
					
						
							|  |  |  |     fprintf(stderr, "Unknown type: %d\n",u); | 
					
						
							|  |  |  |     exit(0); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return(u); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void readswap4(uint32 *buf){ | 
					
						
							|  |  |  |   ((byte *) buf)[0] = cursor[0]; | 
					
						
							|  |  |  |   ((byte *) buf)[1] = cursor[1]; | 
					
						
							|  |  |  |   ((byte *) buf)[2] = cursor[2]; | 
					
						
							|  |  |  |   ((byte *) buf)[3] = cursor[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ( swaporder ) { | 
					
						
							|  |  |  |     if ( inbyteorder == WKBXDR ) { | 
					
						
							|  |  |  |       *buf = (uint32)ntohl((u_long)*buf); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       byte u[4]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       u[0] = ((byte *) buf)[3]; | 
					
						
							|  |  |  |       u[1] = ((byte *) buf)[2]; | 
					
						
							|  |  |  |       u[2] = ((byte *) buf)[1]; | 
					
						
							|  |  |  |       u[3] = ((byte *) buf)[0]; | 
					
						
							|  |  |  |       ((byte *) buf)[0] = u[0]; | 
					
						
							|  |  |  |       ((byte *) buf)[1] = u[1]; | 
					
						
							|  |  |  |       ((byte *) buf)[2] = u[2]; | 
					
						
							|  |  |  |       ((byte *) buf)[3] = u[3]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   cursor += 4; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void readswap8(double *buf) { | 
					
						
							|  |  |  |   ((byte *) buf)[0] = cursor[0]; | 
					
						
							|  |  |  |   ((byte *) buf)[1] = cursor[1]; | 
					
						
							|  |  |  |   ((byte *) buf)[2] = cursor[2]; | 
					
						
							|  |  |  |   ((byte *) buf)[3] = cursor[3]; | 
					
						
							|  |  |  |   ((byte *) buf)[4] = cursor[4]; | 
					
						
							|  |  |  |   ((byte *) buf)[5] = cursor[5]; | 
					
						
							|  |  |  |   ((byte *) buf)[6] = cursor[6]; | 
					
						
							|  |  |  |   ((byte *) buf)[7] = cursor[7]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ( swaporder ) { | 
					
						
							|  |  |  |     if ( inbyteorder == WKBXDR ) { | 
					
						
							|  |  |  |       u_long u[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       u[0] = ((u_long *) buf)[0]; | 
					
						
							|  |  |  |       u[1] = ((u_long *) buf)[1]; | 
					
						
							|  |  |  |       ((u_long *) buf)[1] = ntohl(u[0]); | 
					
						
							|  |  |  |       ((u_long *) buf)[0] = ntohl(u[1]); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       byte u[8]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       u[0] = ((byte *) buf)[7]; | 
					
						
							|  |  |  |       u[1] = ((byte *) buf)[6]; | 
					
						
							|  |  |  |       u[2] = ((byte *) buf)[5]; | 
					
						
							|  |  |  |       u[3] = ((byte *) buf)[4]; | 
					
						
							|  |  |  |       u[4] = ((byte *) buf)[3]; | 
					
						
							|  |  |  |       u[5] = ((byte *) buf)[2]; | 
					
						
							|  |  |  |       u[6] = ((byte *) buf)[1]; | 
					
						
							|  |  |  |       u[7] = ((byte *) buf)[0]; | 
					
						
							|  |  |  |       ((byte *) buf)[0] = u[0]; | 
					
						
							|  |  |  |       ((byte *) buf)[1] = u[1]; | 
					
						
							|  |  |  |       ((byte *) buf)[2] = u[2]; | 
					
						
							|  |  |  |       ((byte *) buf)[3] = u[3]; | 
					
						
							|  |  |  |       ((byte *) buf)[4] = u[4]; | 
					
						
							|  |  |  |       ((byte *) buf)[5] = u[5]; | 
					
						
							|  |  |  |       ((byte *) buf)[6] = u[6]; | 
					
						
							|  |  |  |       ((byte *) buf)[7] = u[7]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   cursor += 8; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Term get_point(char *func){ | 
					
						
							|  |  |  |   Term args[2]; | 
					
						
							|  |  |  |   Functor functor; | 
					
						
							|  |  |  |   double d; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if(func == NULL) | 
					
						
							|  |  |  |     /*functor "," => (_,_)*/ | 
					
						
							|  |  |  |     functor = Yap_MkFunctor(Yap_LookupAtom(","), 2); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     functor = Yap_MkFunctor(Yap_LookupAtom(func), 2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* read the X */ | 
					
						
							|  |  |  |   readswap8(&d); | 
					
						
							|  |  |  |   args[0] = MkFloatTerm(d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* read the Y */ | 
					
						
							|  |  |  |   readswap8(&d); | 
					
						
							|  |  |  |   args[1] = MkFloatTerm(d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return Yap_MkApplTerm(functor, 2, args); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Term get_linestring(char *func){ | 
					
						
							| 
									
										
										
										
											2011-05-09 19:36:51 +01:00
										 |  |  |   CACHE_REGS | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 12:23:29 +00:00
										 |  |  |   Term *c_list; | 
					
						
							|  |  |  |   Term list; | 
					
						
							|  |  |  |   Functor functor; | 
					
						
							|  |  |  |   uint32 n; | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* read the number of vertices */ | 
					
						
							|  |  |  |   readswap4(&n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* space for arguments */ | 
					
						
							|  |  |  |   c_list = (Term *) calloc(sizeof(Term),n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for ( i = 0; i < n; i++) { | 
					
						
							|  |  |  |     c_list[i] = get_point(NULL); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   list = MkAtomTerm(Yap_LookupAtom("[]")); | 
					
						
							|  |  |  |   for (i = n - 1; i >= 0; i--) { | 
					
						
							|  |  |  |     list = MkPairTerm(c_list[i],list); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if(func == NULL) | 
					
						
							|  |  |  |     return list; | 
					
						
							|  |  |  |   else{ | 
					
						
							|  |  |  |     functor = Yap_MkFunctor(Yap_LookupAtom(func), 1); | 
					
						
							|  |  |  |     return Yap_MkApplTerm(functor, 1, &list); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Term get_polygon(char *func){ | 
					
						
							| 
									
										
										
										
											2011-05-09 19:36:51 +01:00
										 |  |  |   CACHE_REGS | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 12:23:29 +00:00
										 |  |  |   uint32 r; | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  |   Functor functor; | 
					
						
							|  |  |  |   Term *c_list; | 
					
						
							|  |  |  |   Term list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* read the number of rings */ | 
					
						
							|  |  |  |   readswap4(&r); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* space for rings */ | 
					
						
							|  |  |  |   c_list = (Term *) calloc(sizeof(Term),r); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for ( i = 0; i < r; i++ ) { | 
					
						
							|  |  |  |     c_list[i] = get_linestring(NULL); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   list = MkAtomTerm(Yap_LookupAtom("[]")); | 
					
						
							|  |  |  |   for (i = r - 1; i >= 0; i--) { | 
					
						
							|  |  |  |     list = MkPairTerm(c_list[i],list); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if(func == NULL) | 
					
						
							|  |  |  |     return list; | 
					
						
							|  |  |  |   else{ | 
					
						
							|  |  |  |     functor = Yap_MkFunctor(Yap_LookupAtom("polygon"), 1); | 
					
						
							|  |  |  |     return Yap_MkApplTerm(functor, 1, &list); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Term get_geometry(uint32 type){ | 
					
						
							| 
									
										
										
										
											2011-05-09 19:36:51 +01:00
										 |  |  |   CACHE_REGS | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 12:23:29 +00:00
										 |  |  |   switch(type) { | 
					
						
							|  |  |  |   case WKBPOINT: | 
					
						
							|  |  |  |     return get_point("point"); | 
					
						
							|  |  |  |   case WKBLINESTRING: | 
					
						
							|  |  |  |     return get_linestring("linestring"); | 
					
						
							|  |  |  |   case WKBPOLYGON: | 
					
						
							|  |  |  |     return get_polygon("polygon"); | 
					
						
							|  |  |  |   case WKBMULTIPOINT: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       byte b; | 
					
						
							|  |  |  |       uint32 n, u; | 
					
						
							|  |  |  |       int i; | 
					
						
							|  |  |  |       Functor functor; | 
					
						
							|  |  |  |       Term *c_list; | 
					
						
							|  |  |  |       Term list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* read the number of points */ | 
					
						
							|  |  |  |       readswap4(&n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* space for points */ | 
					
						
							|  |  |  |       c_list = (Term *) calloc(sizeof(Term),n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for ( i = 0; i < n; i++ ) { | 
					
						
							|  |  |  | 	/* read (and ignore) the byteorder and type */ | 
					
						
							|  |  |  | 	b = get_inbyteorder(); | 
					
						
							|  |  |  | 	u = get_wkbType(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c_list[i] = get_point(NULL); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       list = MkAtomTerm(Yap_LookupAtom("[]")); | 
					
						
							|  |  |  |       for (i = n - 1; i >= 0; i--) { | 
					
						
							|  |  |  | 	list = MkPairTerm(c_list[i],list); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       functor = Yap_MkFunctor(Yap_LookupAtom("multipoint"), 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return Yap_MkApplTerm(functor, 1, &list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case WKBMULTILINESTRING: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       byte b; | 
					
						
							|  |  |  |       uint32 n, u; | 
					
						
							|  |  |  |       int i; | 
					
						
							|  |  |  |       Functor functor; | 
					
						
							|  |  |  |       Term *c_list; | 
					
						
							|  |  |  |       Term list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* read the number of polygons */ | 
					
						
							|  |  |  |       readswap4(&n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* space for polygons*/ | 
					
						
							|  |  |  |       c_list = (Term *) calloc(sizeof(Term),n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for ( i = 0; i < n; i++ ) { | 
					
						
							|  |  |  | 	/* read (and ignore) the byteorder and type */ | 
					
						
							|  |  |  | 	b = get_inbyteorder(); | 
					
						
							|  |  |  | 	u = get_wkbType(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c_list[i] = get_linestring(NULL); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       list = MkAtomTerm(Yap_LookupAtom("[]")); | 
					
						
							|  |  |  |       for (i = n - 1; i >= 0; i--) { | 
					
						
							|  |  |  | 	list = MkPairTerm(c_list[i],list); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       functor = Yap_MkFunctor(Yap_LookupAtom("multilinestring"), 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return Yap_MkApplTerm(functor, 1, &list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case WKBMULTIPOLYGON: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       byte b; | 
					
						
							|  |  |  |       uint32 n, u; | 
					
						
							|  |  |  |       int i; | 
					
						
							|  |  |  |       Functor functor; | 
					
						
							|  |  |  |       Term *c_list; | 
					
						
							|  |  |  |       Term list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* read the number of polygons */ | 
					
						
							|  |  |  |       readswap4(&n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* space for polygons*/ | 
					
						
							|  |  |  |       c_list = (Term *) calloc(sizeof(Term),n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for ( i = 0; i < n; i++ ) { | 
					
						
							|  |  |  | 	/* read (and ignore) the byteorder and type */ | 
					
						
							|  |  |  | 	b = get_inbyteorder(); | 
					
						
							|  |  |  | 	u = get_wkbType(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c_list[i] = get_polygon(NULL); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       list = MkAtomTerm(Yap_LookupAtom("[]")); | 
					
						
							|  |  |  |       for (i = n - 1; i >= 0; i--) { | 
					
						
							|  |  |  | 	list = MkPairTerm(c_list[i],list); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       functor = Yap_MkFunctor(Yap_LookupAtom("multipolygon"), 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return Yap_MkApplTerm(functor, 1, &list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case WKBGEOMETRYCOLLECTION: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       byte b; | 
					
						
							|  |  |  |       uint32 n; | 
					
						
							|  |  |  |       int i; | 
					
						
							|  |  |  |       Functor functor; | 
					
						
							|  |  |  |       Term *c_list; | 
					
						
							|  |  |  |       Term list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* read the number of geometries */ | 
					
						
							|  |  |  |       readswap4(&n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* space for geometries*/ | 
					
						
							|  |  |  |       c_list = (Term *) calloc(sizeof(Term),n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for ( i = 0; i < n; i++ ) { | 
					
						
							|  |  |  | 	b = get_inbyteorder(); | 
					
						
							|  |  |  | 	c_list[i] = get_geometry(get_wkbType()); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       list = MkAtomTerm(Yap_LookupAtom("[]")); | 
					
						
							|  |  |  |       for (i = n - 1; i >= 0; i--) { | 
					
						
							|  |  |  | 	list = MkPairTerm(c_list[i],list); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       functor = Yap_MkFunctor(Yap_LookupAtom("geometrycollection"), 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return Yap_MkApplTerm(functor, 1, &list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return MkAtomTerm(Yap_LookupAtom("[]")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /*MYDDAS_MYSQL*/
 |