Completed basic socket work.
Processes multiple requests concurrently (single thread). Currently just echos received data.
This commit is contained in:
		
							
								
								
									
										121
									
								
								src/main.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								src/main.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,121 @@ | ||||
| /*  | ||||
|  * File:   main.c | ||||
|  * Author: sam | ||||
|  * | ||||
|  * Created on 16 July 2014, 20:05 | ||||
|  */ | ||||
|  | ||||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdbool.h> | ||||
| #include <string.h> | ||||
| #include <strings.h> | ||||
|  | ||||
| #include "ut/utlist.h" | ||||
| #include "main.h" | ||||
| #include "socket.h" | ||||
|  | ||||
| int serverfd = 0; | ||||
|  | ||||
| /* | ||||
|  *  | ||||
|  */ | ||||
| int main(int argc, char** argv) { | ||||
|      | ||||
|     skt_elem *connections = NULL; | ||||
|      | ||||
|     serverfd = svr_create(); | ||||
|     svr_listen(serverfd, 1234); | ||||
|      | ||||
|     while(1) { | ||||
|         //Accept new connections | ||||
|         while(svr_canaccept(serverfd)) { | ||||
|             skt_info *info = svr_accept(serverfd); | ||||
|             if (info != NULL) { | ||||
|                 skt_elem* newconn = calloc(1, sizeof(skt_elem)); | ||||
|                 newconn->info = info; | ||||
|                 LL_APPEND(connections, newconn); | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         skt_elem *elem, *tmp; | ||||
|         //Read from connections | ||||
|         LL_FOREACH(connections, elem) { | ||||
|             if (skt_canread(elem->info)) { | ||||
|                 skt_read(elem->info); | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         //Process sockets | ||||
|         LL_FOREACH(connections, elem) { | ||||
|             if (utstring_len(elem->info->read) > 0) { | ||||
|                 utstring_printf(elem->info->write, "->"); | ||||
|                 utstring_concat(elem->info->write, elem->info->read); | ||||
|                 utstring_clear(elem->info->read); | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         //Write to connections | ||||
|         LL_FOREACH(connections, elem) { | ||||
|             if (utstring_len(elem->info->write) > 0) { | ||||
|                 skt_write(elem->info); | ||||
|             }  | ||||
|         } | ||||
|          | ||||
|         time_t current = time(NULL); | ||||
|         time_t timeout = 5; | ||||
|         time_t maxlife = 500; | ||||
|         //Close where needed | ||||
|         LL_FOREACH(connections, elem) { | ||||
|             if (current - elem->info->last_act > timeout) { | ||||
|                 info("[#%lu %s] Timeout", elem->info->id, skt_addr(elem->info)); | ||||
|                 elem->info->close = true; | ||||
|             } | ||||
|             if (current - elem->info->time_opened> maxlife) { | ||||
|                 info("[#%lu %s] Reached max life", elem->info->id, skt_addr(elem->info)); | ||||
|                 elem->info->close = true; | ||||
|             } | ||||
|             if (elem->info->close == true) { | ||||
|                 skt_close(elem->info); | ||||
|             } | ||||
|         } | ||||
|         //Delete closed connections | ||||
|         LL_FOREACH_SAFE(connections, elem, tmp) { | ||||
|             if (elem->info->closed) { | ||||
|                 LL_DELETE(connections, elem); | ||||
|                 skt_delete(elem->info); | ||||
|                 free(elem); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     svr_release(serverfd); | ||||
|     serverfd = 0; | ||||
|      | ||||
|     return (EXIT_SUCCESS); | ||||
| } | ||||
|  | ||||
| void fatal(char* msg) { | ||||
|     fprintf(stderr, "\n"); | ||||
|     perror(msg); | ||||
|     exit(EXIT_FAILURE); | ||||
| } | ||||
| void warning(char* msg, bool showPError) { | ||||
|      | ||||
|     char warning[512]; | ||||
|     bzero(&warning, sizeof warning); | ||||
|     snprintf(warning, 511, "Warning: %s", msg); | ||||
|      | ||||
|     if (showPError == true) { | ||||
|         perror(warning); | ||||
|     } else { | ||||
|         fprintf(stderr, "%s\n", warning); | ||||
|     } | ||||
| } | ||||
| void info(char* msg, ...) { | ||||
|     va_list va; | ||||
|     va_start(va, msg); | ||||
|     vfprintf(stdout, msg, va); | ||||
|     fputc('\n', stdout); | ||||
|     va_end(va); | ||||
| } | ||||
							
								
								
									
										29
									
								
								src/main.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								src/main.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| /*  | ||||
|  * File:   main.h | ||||
|  * Author: sam | ||||
|  * | ||||
|  * Created on 16 July 2014, 20:18 | ||||
|  */ | ||||
|  | ||||
| #ifndef MAIN_H | ||||
| #define	MAIN_H | ||||
|  | ||||
| #ifdef	__cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
|  | ||||
| #include <stdbool.h> | ||||
|      | ||||
| int main(int argc, char** argv); | ||||
|  | ||||
| void fatal(char* msg); | ||||
| void warning(char* msg, bool showPError); | ||||
| void info(char* msg, ...); | ||||
|  | ||||
|  | ||||
| #ifdef	__cplusplus | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #endif	/* MAIN_H */ | ||||
|  | ||||
							
								
								
									
										174
									
								
								src/socket.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								src/socket.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | ||||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <unistd.h> | ||||
| #include <sys/types.h> | ||||
| #include <sys/socket.h> | ||||
| #include <netinet/in.h> | ||||
| #include <stdbool.h> | ||||
| #include <time.h> | ||||
| #include <sys/ioctl.h> | ||||
| #include <errno.h> | ||||
| #include <poll.h> | ||||
| #include <arpa/inet.h> | ||||
| #include "socket.h" | ||||
| #include "ut/utstring.h" | ||||
| #include "main.h" | ||||
|  | ||||
| u_int64_t skt_nextid() { | ||||
|     static u_int64_t id = 0; | ||||
|     return id++; | ||||
| } | ||||
| skt_info* skt_new(int fd) { | ||||
|     skt_info* skt = calloc(1, sizeof(skt_info)); | ||||
|     skt->id = skt_nextid(); | ||||
|     skt->fd = fd; | ||||
|     skt->last_act = skt->time_opened = time(NULL); | ||||
|     utstring_new(skt->read); | ||||
|     utstring_new(skt->write); | ||||
|     skt->close = false; | ||||
|     skt->closed = false; | ||||
|      | ||||
|     return skt; | ||||
| } | ||||
| void skt_delete(skt_info* skt) { | ||||
|     utstring_free(skt->read); | ||||
|     utstring_free(skt->write); | ||||
|     free(skt->clientaddr); | ||||
|     free(skt); | ||||
| } | ||||
|  | ||||
| bool skt_canread(skt_info* skt) { | ||||
|     int len = 0; | ||||
|     if (ioctl(skt->fd, FIONREAD, &len) < 0) { | ||||
|         warning("ioctl failed", true); | ||||
|         return false; | ||||
|     } | ||||
|     return len > 0; | ||||
| } | ||||
| uint32_t skt_read(skt_info* skt) { | ||||
|     char buffer[1024]; | ||||
|     memset(buffer, 0, 1024); | ||||
|      | ||||
|     int result = read(skt->fd, &buffer,1023); | ||||
|     if (result < 0 && errno != EAGAIN && errno != EWOULDBLOCK) { | ||||
|         warning("read error", true); | ||||
|         skt->close = true; | ||||
|         return 0; | ||||
|     } | ||||
|     skt->last_act = time(NULL); | ||||
|     utstring_printf(skt->read, "%s", buffer); | ||||
|     return result; //Number of bytes read | ||||
| } | ||||
| uint32_t skt_write(skt_info* skt) { | ||||
|     if (utstring_len(skt->write) == 0) { | ||||
|         return 0; | ||||
|     } | ||||
|      | ||||
|     int result = write(skt->fd, utstring_body(skt->write), utstring_len(skt->write)); | ||||
|     if (result < 0 && errno != EAGAIN && errno != EWOULDBLOCK) { | ||||
|         warning("write error", true); | ||||
|         skt->close = true; | ||||
|         return 0; | ||||
|     } | ||||
|     skt->last_act = time(NULL); | ||||
|      | ||||
|     if (result == utstring_len(skt->write)) { | ||||
|         utstring_free(skt->write); | ||||
|         utstring_new(skt->write); | ||||
|         return 0; | ||||
|     } | ||||
|     //remove first x chars | ||||
|     char* newstr = calloc(utstring_len(skt->write) - result + 1, sizeof(char)); | ||||
|      | ||||
|     char* writeBody = utstring_body(skt->write); | ||||
|     strcpy(newstr, writeBody + (sizeof(char) * result)); | ||||
|      | ||||
|     utstring_free(skt->write); | ||||
|     utstring_new(skt->write); | ||||
|     utstring_printf(skt->write, "%s", newstr); | ||||
|     free(newstr); | ||||
|     return result; //bytes written | ||||
| } | ||||
| void skt_close(skt_info* skt) { | ||||
|     if (skt->closed == true) { | ||||
|         return; | ||||
|     } | ||||
|     info("[#%lu %s] Closed", skt->id, skt_addr(skt)); | ||||
|     if (close(skt->fd) < 0) { | ||||
|         warning("error closing socket", true); | ||||
|     } | ||||
|     skt->closed = true; | ||||
| } | ||||
| char* skt_addr(skt_info *skt) { | ||||
|     char* address = inet_ntoa(skt->clientaddr->sin_addr); | ||||
|     return address; | ||||
| } | ||||
|  | ||||
| int svr_create() { | ||||
|     int fd = 0; | ||||
|     fd = socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0); | ||||
|     if (fd < 0) { | ||||
|         fatal("could not create socket"); | ||||
|     } | ||||
|     return fd; | ||||
| } | ||||
| void svr_listen(int fd, uint16_t port) { | ||||
|     struct sockaddr_in server_address; | ||||
|     memset(&server_address, 0, sizeof server_address); | ||||
|     server_address.sin_family = AF_INET; | ||||
|     server_address.sin_addr.s_addr = INADDR_ANY; | ||||
|     server_address.sin_port = htons(port); | ||||
|      | ||||
|     if (bind(fd, (struct sockaddr*)&server_address, sizeof server_address) < 0) { | ||||
|         warning("failed to bind", true); | ||||
|         close(fd); | ||||
|         exit(EXIT_FAILURE); | ||||
|     } | ||||
|     if (listen(fd, 16) < 0) { | ||||
|         warning("failed to listen", true); | ||||
|         close(fd); | ||||
|         exit(EXIT_FAILURE); | ||||
|     } | ||||
|     info("Listening on port %u", port); | ||||
| } | ||||
| void svr_release(int fd) { | ||||
|     if (close(fd) < 0) { | ||||
|         warning("could not close socket", true); | ||||
|     } | ||||
| } | ||||
| bool svr_canaccept(int fd) { | ||||
|     struct pollfd* pfd = calloc(1, sizeof(struct pollfd)); | ||||
|      | ||||
|     pfd[0].fd = fd; | ||||
|     pfd[0].events = POLLIN; | ||||
|      | ||||
|     if (poll(pfd, 1, 50/*ms*/) < 0) { | ||||
|         warning("poll failed", true); | ||||
|         return false; | ||||
|     } | ||||
|     if ((pfd[0].revents & POLLIN) == POLLIN) { | ||||
|         return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
| skt_info* svr_accept(int fd) { | ||||
|     struct sockaddr_in* clientaddr = calloc(1, sizeof(struct sockaddr_in)); | ||||
|      | ||||
|     int clientfd=0; | ||||
|     socklen_t clientaddr_len = (socklen_t)sizeof(struct sockaddr_in); | ||||
|     clientfd = accept(fd, (struct sockaddr*)clientaddr, &clientaddr_len); | ||||
|     if (clientfd < 0 && errno != EAGAIN && errno != EWOULDBLOCK) { | ||||
|         warning("error accepting connection", true); | ||||
|         return NULL; | ||||
|     } | ||||
|      | ||||
|     skt_info* skt = skt_new(clientfd); | ||||
|     skt->clientaddr = clientaddr; | ||||
|     skt->fd = clientfd; | ||||
|      | ||||
|     info("[#%lu %s] New Connection", skt->id, skt_addr(skt)); | ||||
|      | ||||
|     return skt; | ||||
|      | ||||
| } | ||||
							
								
								
									
										64
									
								
								src/socket.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								src/socket.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| /*  | ||||
|  * File:   socket.h | ||||
|  * Author: sam | ||||
|  * | ||||
|  * Created on 16 July 2014, 20:19 | ||||
|  */ | ||||
|  | ||||
| #ifndef SOCKET_H | ||||
| #define	SOCKET_H | ||||
|  | ||||
| #ifdef	__cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
|  | ||||
| #include <stdbool.h> | ||||
| #include <sys/types.h> | ||||
| #include <sys/socket.h> | ||||
| #include <netinet/in.h> | ||||
| #include <time.h> | ||||
| #include "ut/utstring.h" | ||||
|       | ||||
|     typedef struct skt_info skt_info; | ||||
|      | ||||
|     struct skt_info { | ||||
|         u_int64_t id; | ||||
|         int fd; | ||||
|         time_t time_opened; | ||||
|         time_t last_act; | ||||
|         UT_string *read; | ||||
|         UT_string *write; | ||||
|         bool close; | ||||
|         bool closed; | ||||
|         struct sockaddr_in* clientaddr; | ||||
|     }; | ||||
|      | ||||
|     typedef struct skt_elem { | ||||
|         skt_info* info; | ||||
|         struct skt_elem *next; | ||||
|     } skt_elem; | ||||
|      | ||||
|     u_int64_t skt_nextid(); | ||||
|     skt_info* skt_new(int fd); | ||||
|     void skt_delete(skt_info *skt); | ||||
|      | ||||
|     bool skt_canread(skt_info *skt); | ||||
|     uint32_t skt_read(skt_info *skt); | ||||
|     uint32_t skt_write(skt_info *skt); | ||||
|     void skt_close(skt_info *skt); | ||||
|     char* skt_addr(skt_info *skt); | ||||
|      | ||||
|     int svr_create(); | ||||
|     void svr_listen(int fd, uint16_t port); | ||||
|     void svr_release(int fd); | ||||
|     bool svr_canaccept(int fd); | ||||
|     skt_info* svr_accept(int fd); | ||||
|      | ||||
|  | ||||
|  | ||||
| #ifdef	__cplusplus | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #endif	/* SOCKET_H */ | ||||
|  | ||||
							
								
								
									
										232
									
								
								src/ut/utarray.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										232
									
								
								src/ut/utarray.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,232 @@ | ||||
| /* | ||||
| Copyright (c) 2008-2014, Troy D. Hanson   http://troydhanson.github.com/uthash/ | ||||
| All rights reserved. | ||||
|  | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions are met: | ||||
|  | ||||
|     * Redistributions of source code must retain the above copyright | ||||
|       notice, this list of conditions and the following disclaimer. | ||||
|  | ||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS | ||||
| IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED | ||||
| TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A | ||||
| PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER | ||||
| OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | ||||
| EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | ||||
| PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | ||||
| PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF | ||||
| LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | ||||
| NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||||
| SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
| */ | ||||
|  | ||||
| /* a dynamic array implementation using macros  | ||||
|  */ | ||||
| #ifndef UTARRAY_H | ||||
| #define UTARRAY_H | ||||
|  | ||||
| #define UTARRAY_VERSION 1.9.9 | ||||
|  | ||||
| #ifdef __GNUC__ | ||||
| #define _UNUSED_ __attribute__ ((__unused__))  | ||||
| #else | ||||
| #define _UNUSED_  | ||||
| #endif | ||||
|  | ||||
| #include <stddef.h>  /* size_t */ | ||||
| #include <string.h>  /* memset, etc */ | ||||
| #include <stdlib.h>  /* exit */ | ||||
|  | ||||
| #define oom() exit(-1) | ||||
|  | ||||
| typedef void (ctor_f)(void *dst, const void *src); | ||||
| typedef void (dtor_f)(void *elt); | ||||
| typedef void (init_f)(void *elt); | ||||
| typedef struct { | ||||
|     size_t sz; | ||||
|     init_f *init; | ||||
|     ctor_f *copy; | ||||
|     dtor_f *dtor; | ||||
| } UT_icd; | ||||
|  | ||||
| typedef struct { | ||||
|     unsigned i,n;/* i: index of next available slot, n: num slots */ | ||||
|     UT_icd icd;  /* initializer, copy and destructor functions */ | ||||
|     char *d;     /* n slots of size icd->sz*/ | ||||
| } UT_array; | ||||
|  | ||||
| #define utarray_init(a,_icd) do {                                             \ | ||||
|   memset(a,0,sizeof(UT_array));                                               \ | ||||
|   (a)->icd=*_icd;                                                             \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_done(a) do {                                                  \ | ||||
|   if ((a)->n) {                                                               \ | ||||
|     if ((a)->icd.dtor) {                                                      \ | ||||
|       size_t _ut_i;                                                           \ | ||||
|       for(_ut_i=0; _ut_i < (a)->i; _ut_i++) {                                 \ | ||||
|         (a)->icd.dtor(utarray_eltptr(a,_ut_i));                               \ | ||||
|       }                                                                       \ | ||||
|     }                                                                         \ | ||||
|     free((a)->d);                                                             \ | ||||
|   }                                                                           \ | ||||
|   (a)->n=0;                                                                   \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_new(a,_icd) do {                                              \ | ||||
|   a=(UT_array*)malloc(sizeof(UT_array));                                      \ | ||||
|   utarray_init(a,_icd);                                                       \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_free(a) do {                                                  \ | ||||
|   utarray_done(a);                                                            \ | ||||
|   free(a);                                                                    \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_reserve(a,by) do {                                            \ | ||||
|   if (((a)->i+by) > ((a)->n)) {                                               \ | ||||
|     while(((a)->i+by) > ((a)->n)) { (a)->n = ((a)->n ? (2*(a)->n) : 8); }     \ | ||||
|     if ( ((a)->d=(char*)realloc((a)->d, (a)->n*(a)->icd.sz)) == NULL) oom();  \ | ||||
|   }                                                                           \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_push_back(a,p) do {                                           \ | ||||
|   utarray_reserve(a,1);                                                       \ | ||||
|   if ((a)->icd.copy) { (a)->icd.copy( _utarray_eltptr(a,(a)->i++), p); }      \ | ||||
|   else { memcpy(_utarray_eltptr(a,(a)->i++), p, (a)->icd.sz); };              \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_pop_back(a) do {                                              \ | ||||
|   if ((a)->icd.dtor) { (a)->icd.dtor( _utarray_eltptr(a,--((a)->i))); }       \ | ||||
|   else { (a)->i--; }                                                          \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_extend_back(a) do {                                           \ | ||||
|   utarray_reserve(a,1);                                                       \ | ||||
|   if ((a)->icd.init) { (a)->icd.init(_utarray_eltptr(a,(a)->i)); }            \ | ||||
|   else { memset(_utarray_eltptr(a,(a)->i),0,(a)->icd.sz); }                   \ | ||||
|   (a)->i++;                                                                   \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_len(a) ((a)->i) | ||||
|  | ||||
| #define utarray_eltptr(a,j) (((j) < (a)->i) ? _utarray_eltptr(a,j) : NULL) | ||||
| #define _utarray_eltptr(a,j) ((char*)((a)->d + ((a)->icd.sz*(j) ))) | ||||
|  | ||||
| #define utarray_insert(a,p,j) do {                                            \ | ||||
|   if (j > (a)->i) utarray_resize(a,j);                                        \ | ||||
|   utarray_reserve(a,1);                                                       \ | ||||
|   if ((j) < (a)->i) {                                                         \ | ||||
|     memmove( _utarray_eltptr(a,(j)+1), _utarray_eltptr(a,j),                  \ | ||||
|              ((a)->i - (j))*((a)->icd.sz));                                   \ | ||||
|   }                                                                           \ | ||||
|   if ((a)->icd.copy) { (a)->icd.copy( _utarray_eltptr(a,j), p); }             \ | ||||
|   else { memcpy(_utarray_eltptr(a,j), p, (a)->icd.sz); };                     \ | ||||
|   (a)->i++;                                                                   \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_inserta(a,w,j) do {                                           \ | ||||
|   if (utarray_len(w) == 0) break;                                             \ | ||||
|   if (j > (a)->i) utarray_resize(a,j);                                        \ | ||||
|   utarray_reserve(a,utarray_len(w));                                          \ | ||||
|   if ((j) < (a)->i) {                                                         \ | ||||
|     memmove(_utarray_eltptr(a,(j)+utarray_len(w)),                            \ | ||||
|             _utarray_eltptr(a,j),                                             \ | ||||
|             ((a)->i - (j))*((a)->icd.sz));                                    \ | ||||
|   }                                                                           \ | ||||
|   if ((a)->icd.copy) {                                                        \ | ||||
|     size_t _ut_i;                                                             \ | ||||
|     for(_ut_i=0;_ut_i<(w)->i;_ut_i++) {                                       \ | ||||
|       (a)->icd.copy(_utarray_eltptr(a,j+_ut_i), _utarray_eltptr(w,_ut_i));    \ | ||||
|     }                                                                         \ | ||||
|   } else {                                                                    \ | ||||
|     memcpy(_utarray_eltptr(a,j), _utarray_eltptr(w,0),                        \ | ||||
|            utarray_len(w)*((a)->icd.sz));                                     \ | ||||
|   }                                                                           \ | ||||
|   (a)->i += utarray_len(w);                                                   \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_resize(dst,num) do {                                          \ | ||||
|   size_t _ut_i;                                                               \ | ||||
|   if (dst->i > (size_t)(num)) {                                               \ | ||||
|     if ((dst)->icd.dtor) {                                                    \ | ||||
|       for(_ut_i=num; _ut_i < dst->i; _ut_i++) {                               \ | ||||
|         (dst)->icd.dtor(utarray_eltptr(dst,_ut_i));                           \ | ||||
|       }                                                                       \ | ||||
|     }                                                                         \ | ||||
|   } else if (dst->i < (size_t)(num)) {                                        \ | ||||
|     utarray_reserve(dst,num-dst->i);                                          \ | ||||
|     if ((dst)->icd.init) {                                                    \ | ||||
|       for(_ut_i=dst->i; _ut_i < num; _ut_i++) {                               \ | ||||
|         (dst)->icd.init(utarray_eltptr(dst,_ut_i));                           \ | ||||
|       }                                                                       \ | ||||
|     } else {                                                                  \ | ||||
|       memset(_utarray_eltptr(dst,dst->i),0,(dst)->icd.sz*(num-dst->i));       \ | ||||
|     }                                                                         \ | ||||
|   }                                                                           \ | ||||
|   dst->i = num;                                                               \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_concat(dst,src) do {                                          \ | ||||
|   utarray_inserta((dst),(src),utarray_len(dst));                              \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_erase(a,pos,len) do {                                         \ | ||||
|   if ((a)->icd.dtor) {                                                        \ | ||||
|     size_t _ut_i;                                                             \ | ||||
|     for(_ut_i=0; _ut_i < len; _ut_i++) {                                      \ | ||||
|       (a)->icd.dtor(utarray_eltptr((a),pos+_ut_i));                           \ | ||||
|     }                                                                         \ | ||||
|   }                                                                           \ | ||||
|   if ((a)->i > (pos+len)) {                                                   \ | ||||
|     memmove( _utarray_eltptr((a),pos), _utarray_eltptr((a),pos+len),          \ | ||||
|             (((a)->i)-(pos+len))*((a)->icd.sz));                              \ | ||||
|   }                                                                           \ | ||||
|   (a)->i -= (len);                                                            \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_renew(a,u) do {                                               \ | ||||
|   if (a) utarray_clear(a); \ | ||||
|   else utarray_new((a),(u));   \ | ||||
| } while(0)  | ||||
|  | ||||
| #define utarray_clear(a) do {                                                 \ | ||||
|   if ((a)->i > 0) {                                                           \ | ||||
|     if ((a)->icd.dtor) {                                                      \ | ||||
|       size_t _ut_i;                                                           \ | ||||
|       for(_ut_i=0; _ut_i < (a)->i; _ut_i++) {                                 \ | ||||
|         (a)->icd.dtor(utarray_eltptr(a,_ut_i));                               \ | ||||
|       }                                                                       \ | ||||
|     }                                                                         \ | ||||
|     (a)->i = 0;                                                               \ | ||||
|   }                                                                           \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_sort(a,cmp) do {                                              \ | ||||
|   qsort((a)->d, (a)->i, (a)->icd.sz, cmp);                                    \ | ||||
| } while(0) | ||||
|  | ||||
| #define utarray_find(a,v,cmp) bsearch((v),(a)->d,(a)->i,(a)->icd.sz,cmp) | ||||
|  | ||||
| #define utarray_front(a) (((a)->i) ? (_utarray_eltptr(a,0)) : NULL) | ||||
| #define utarray_next(a,e) (((e)==NULL) ? utarray_front(a) : ((((a)->i) > (utarray_eltidx(a,e)+1)) ? _utarray_eltptr(a,utarray_eltidx(a,e)+1) : NULL)) | ||||
| #define utarray_prev(a,e) (((e)==NULL) ? utarray_back(a) : ((utarray_eltidx(a,e) > 0) ? _utarray_eltptr(a,utarray_eltidx(a,e)-1) : NULL)) | ||||
| #define utarray_back(a) (((a)->i) ? (_utarray_eltptr(a,(a)->i-1)) : NULL) | ||||
| #define utarray_eltidx(a,e) (((char*)(e) >= (char*)((a)->d)) ? (((char*)(e) - (char*)((a)->d))/(size_t)(a)->icd.sz) : -1) | ||||
|  | ||||
| /* last we pre-define a few icd for common utarrays of ints and strings */ | ||||
| static void utarray_str_cpy(void *dst, const void *src) { | ||||
|   char **_src = (char**)src, **_dst = (char**)dst; | ||||
|   *_dst = (*_src == NULL) ? NULL : strdup(*_src); | ||||
| } | ||||
| static void utarray_str_dtor(void *elt) { | ||||
|   char **eltc = (char**)elt; | ||||
|   if (*eltc) free(*eltc); | ||||
| } | ||||
| static const UT_icd ut_str_icd _UNUSED_ = {sizeof(char*),NULL,utarray_str_cpy,utarray_str_dtor}; | ||||
| static const UT_icd ut_int_icd _UNUSED_ = {sizeof(int),NULL,NULL,NULL}; | ||||
| static const UT_icd ut_ptr_icd _UNUSED_ = {sizeof(void*),NULL,NULL,NULL}; | ||||
|  | ||||
|  | ||||
| #endif /* UTARRAY_H */ | ||||
							
								
								
									
										949
									
								
								src/ut/uthash.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										949
									
								
								src/ut/uthash.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,949 @@ | ||||
| /* | ||||
| Copyright (c) 2003-2014, Troy D. Hanson     http://troydhanson.github.com/uthash/ | ||||
| All rights reserved. | ||||
|  | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions are met: | ||||
|  | ||||
|     * Redistributions of source code must retain the above copyright | ||||
|       notice, this list of conditions and the following disclaimer. | ||||
|  | ||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS | ||||
| IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED | ||||
| TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A | ||||
| PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER | ||||
| OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | ||||
| EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | ||||
| PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | ||||
| PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF | ||||
| LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | ||||
| NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||||
| SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
| */ | ||||
|  | ||||
| #ifndef UTHASH_H | ||||
| #define UTHASH_H  | ||||
|  | ||||
| #include <string.h>   /* memcmp,strlen */ | ||||
| #include <stddef.h>   /* ptrdiff_t */ | ||||
| #include <stdlib.h>   /* exit() */ | ||||
|  | ||||
| /* These macros use decltype or the earlier __typeof GNU extension. | ||||
|    As decltype is only available in newer compilers (VS2010 or gcc 4.3+ | ||||
|    when compiling c++ source) this code uses whatever method is needed | ||||
|    or, for VS2008 where neither is available, uses casting workarounds. */ | ||||
| #ifdef _MSC_VER         /* MS compiler */ | ||||
| #if _MSC_VER >= 1600 && defined(__cplusplus)  /* VS2010 or newer in C++ mode */ | ||||
| #define DECLTYPE(x) (decltype(x)) | ||||
| #else                   /* VS2008 or older (or VS2010 in C mode) */ | ||||
| #define NO_DECLTYPE | ||||
| #define DECLTYPE(x) | ||||
| #endif | ||||
| #else                   /* GNU, Sun and other compilers */ | ||||
| #define DECLTYPE(x) (__typeof(x)) | ||||
| #endif | ||||
|  | ||||
| #ifdef NO_DECLTYPE | ||||
| #define DECLTYPE_ASSIGN(dst,src)                                                 \ | ||||
| do {                                                                             \ | ||||
|   char **_da_dst = (char**)(&(dst));                                             \ | ||||
|   *_da_dst = (char*)(src);                                                       \ | ||||
| } while(0) | ||||
| #else  | ||||
| #define DECLTYPE_ASSIGN(dst,src)                                                 \ | ||||
| do {                                                                             \ | ||||
|   (dst) = DECLTYPE(dst)(src);                                                    \ | ||||
| } while(0) | ||||
| #endif | ||||
|  | ||||
| /* a number of the hash function use uint32_t which isn't defined on win32 */ | ||||
| #ifdef _MSC_VER | ||||
| typedef unsigned int uint32_t; | ||||
| typedef unsigned char uint8_t; | ||||
| #else | ||||
| #include <inttypes.h>   /* uint32_t */ | ||||
| #endif | ||||
|  | ||||
| #define UTHASH_VERSION 1.9.9 | ||||
|  | ||||
| #ifndef uthash_fatal | ||||
| #define uthash_fatal(msg) exit(-1)        /* fatal error (out of memory,etc) */ | ||||
| #endif | ||||
| #ifndef uthash_malloc | ||||
| #define uthash_malloc(sz) malloc(sz)      /* malloc fcn                      */ | ||||
| #endif | ||||
| #ifndef uthash_free | ||||
| #define uthash_free(ptr,sz) free(ptr)     /* free fcn                        */ | ||||
| #endif | ||||
|  | ||||
| #ifndef uthash_noexpand_fyi | ||||
| #define uthash_noexpand_fyi(tbl)          /* can be defined to log noexpand  */ | ||||
| #endif | ||||
| #ifndef uthash_expand_fyi | ||||
| #define uthash_expand_fyi(tbl)            /* can be defined to log expands   */ | ||||
| #endif | ||||
|  | ||||
| /* initial number of buckets */ | ||||
| #define HASH_INITIAL_NUM_BUCKETS 32      /* initial number of buckets        */ | ||||
| #define HASH_INITIAL_NUM_BUCKETS_LOG2 5  /* lg2 of initial number of buckets */ | ||||
| #define HASH_BKT_CAPACITY_THRESH 10      /* expand when bucket count reaches */ | ||||
|  | ||||
| /* calculate the element whose hash handle address is hhe */ | ||||
| #define ELMT_FROM_HH(tbl,hhp) ((void*)(((char*)(hhp)) - ((tbl)->hho))) | ||||
|  | ||||
| #define HASH_FIND(hh,head,keyptr,keylen,out)                                     \ | ||||
| do {                                                                             \ | ||||
|   unsigned _hf_bkt,_hf_hashv;                                                    \ | ||||
|   out=NULL;                                                                      \ | ||||
|   if (head) {                                                                    \ | ||||
|      HASH_FCN(keyptr,keylen, (head)->hh.tbl->num_buckets, _hf_hashv, _hf_bkt);   \ | ||||
|      if (HASH_BLOOM_TEST((head)->hh.tbl, _hf_hashv)) {                           \ | ||||
|        HASH_FIND_IN_BKT((head)->hh.tbl, hh, (head)->hh.tbl->buckets[ _hf_bkt ],  \ | ||||
|                         keyptr,keylen,out);                                      \ | ||||
|      }                                                                           \ | ||||
|   }                                                                              \ | ||||
| } while (0) | ||||
|  | ||||
| #ifdef HASH_BLOOM | ||||
| #define HASH_BLOOM_BITLEN (1ULL << HASH_BLOOM) | ||||
| #define HASH_BLOOM_BYTELEN (HASH_BLOOM_BITLEN/8) + ((HASH_BLOOM_BITLEN%8) ? 1:0) | ||||
| #define HASH_BLOOM_MAKE(tbl)                                                     \ | ||||
| do {                                                                             \ | ||||
|   (tbl)->bloom_nbits = HASH_BLOOM;                                               \ | ||||
|   (tbl)->bloom_bv = (uint8_t*)uthash_malloc(HASH_BLOOM_BYTELEN);                 \ | ||||
|   if (!((tbl)->bloom_bv))  { uthash_fatal( "out of memory"); }                   \ | ||||
|   memset((tbl)->bloom_bv, 0, HASH_BLOOM_BYTELEN);                                \ | ||||
|   (tbl)->bloom_sig = HASH_BLOOM_SIGNATURE;                                       \ | ||||
| } while (0)  | ||||
|  | ||||
| #define HASH_BLOOM_FREE(tbl)                                                     \ | ||||
| do {                                                                             \ | ||||
|   uthash_free((tbl)->bloom_bv, HASH_BLOOM_BYTELEN);                              \ | ||||
| } while (0)  | ||||
|  | ||||
| #define HASH_BLOOM_BITSET(bv,idx) (bv[(idx)/8] |= (1U << ((idx)%8))) | ||||
| #define HASH_BLOOM_BITTEST(bv,idx) (bv[(idx)/8] & (1U << ((idx)%8))) | ||||
|  | ||||
| #define HASH_BLOOM_ADD(tbl,hashv)                                                \ | ||||
|   HASH_BLOOM_BITSET((tbl)->bloom_bv, (hashv & (uint32_t)((1ULL << (tbl)->bloom_nbits) - 1))) | ||||
|  | ||||
| #define HASH_BLOOM_TEST(tbl,hashv)                                               \ | ||||
|   HASH_BLOOM_BITTEST((tbl)->bloom_bv, (hashv & (uint32_t)((1ULL << (tbl)->bloom_nbits) - 1))) | ||||
|  | ||||
| #else | ||||
| #define HASH_BLOOM_MAKE(tbl)  | ||||
| #define HASH_BLOOM_FREE(tbl)  | ||||
| #define HASH_BLOOM_ADD(tbl,hashv)  | ||||
| #define HASH_BLOOM_TEST(tbl,hashv) (1) | ||||
| #define HASH_BLOOM_BYTELEN 0 | ||||
| #endif | ||||
|  | ||||
| #define HASH_MAKE_TABLE(hh,head)                                                 \ | ||||
| do {                                                                             \ | ||||
|   (head)->hh.tbl = (UT_hash_table*)uthash_malloc(                                \ | ||||
|                   sizeof(UT_hash_table));                                        \ | ||||
|   if (!((head)->hh.tbl))  { uthash_fatal( "out of memory"); }                    \ | ||||
|   memset((head)->hh.tbl, 0, sizeof(UT_hash_table));                              \ | ||||
|   (head)->hh.tbl->tail = &((head)->hh);                                          \ | ||||
|   (head)->hh.tbl->num_buckets = HASH_INITIAL_NUM_BUCKETS;                        \ | ||||
|   (head)->hh.tbl->log2_num_buckets = HASH_INITIAL_NUM_BUCKETS_LOG2;              \ | ||||
|   (head)->hh.tbl->hho = (char*)(&(head)->hh) - (char*)(head);                    \ | ||||
|   (head)->hh.tbl->buckets = (UT_hash_bucket*)uthash_malloc(                      \ | ||||
|           HASH_INITIAL_NUM_BUCKETS*sizeof(struct UT_hash_bucket));               \ | ||||
|   if (! (head)->hh.tbl->buckets) { uthash_fatal( "out of memory"); }             \ | ||||
|   memset((head)->hh.tbl->buckets, 0,                                             \ | ||||
|           HASH_INITIAL_NUM_BUCKETS*sizeof(struct UT_hash_bucket));               \ | ||||
|   HASH_BLOOM_MAKE((head)->hh.tbl);                                               \ | ||||
|   (head)->hh.tbl->signature = HASH_SIGNATURE;                                    \ | ||||
| } while(0) | ||||
|  | ||||
| #define HASH_ADD(hh,head,fieldname,keylen_in,add)                                \ | ||||
|         HASH_ADD_KEYPTR(hh,head,&((add)->fieldname),keylen_in,add) | ||||
|  | ||||
| #define HASH_REPLACE(hh,head,fieldname,keylen_in,add,replaced)                   \ | ||||
| do {                                                                             \ | ||||
|   replaced=NULL;                                                                 \ | ||||
|   HASH_FIND(hh,head,&((add)->fieldname),keylen_in,replaced);                     \ | ||||
|   if (replaced!=NULL) {                                                          \ | ||||
|      HASH_DELETE(hh,head,replaced);                                              \ | ||||
|   };                                                                             \ | ||||
|   HASH_ADD(hh,head,fieldname,keylen_in,add);                                     \ | ||||
| } while(0) | ||||
|   | ||||
| #define HASH_ADD_KEYPTR(hh,head,keyptr,keylen_in,add)                            \ | ||||
| do {                                                                             \ | ||||
|  unsigned _ha_bkt;                                                               \ | ||||
|  (add)->hh.next = NULL;                                                          \ | ||||
|  (add)->hh.key = (char*)(keyptr);                                                \ | ||||
|  (add)->hh.keylen = (unsigned)(keylen_in);                                       \ | ||||
|  if (!(head)) {                                                                  \ | ||||
|     head = (add);                                                                \ | ||||
|     (head)->hh.prev = NULL;                                                      \ | ||||
|     HASH_MAKE_TABLE(hh,head);                                                    \ | ||||
|  } else {                                                                        \ | ||||
|     (head)->hh.tbl->tail->next = (add);                                          \ | ||||
|     (add)->hh.prev = ELMT_FROM_HH((head)->hh.tbl, (head)->hh.tbl->tail);         \ | ||||
|     (head)->hh.tbl->tail = &((add)->hh);                                         \ | ||||
|  }                                                                               \ | ||||
|  (head)->hh.tbl->num_items++;                                                    \ | ||||
|  (add)->hh.tbl = (head)->hh.tbl;                                                 \ | ||||
|  HASH_FCN(keyptr,keylen_in, (head)->hh.tbl->num_buckets,                         \ | ||||
|          (add)->hh.hashv, _ha_bkt);                                              \ | ||||
|  HASH_ADD_TO_BKT((head)->hh.tbl->buckets[_ha_bkt],&(add)->hh);                   \ | ||||
|  HASH_BLOOM_ADD((head)->hh.tbl,(add)->hh.hashv);                                 \ | ||||
|  HASH_EMIT_KEY(hh,head,keyptr,keylen_in);                                        \ | ||||
|  HASH_FSCK(hh,head);                                                             \ | ||||
| } while(0) | ||||
|  | ||||
| #define HASH_TO_BKT( hashv, num_bkts, bkt )                                      \ | ||||
| do {                                                                             \ | ||||
|   bkt = ((hashv) & ((num_bkts) - 1));                                            \ | ||||
| } while(0) | ||||
|  | ||||
| /* delete "delptr" from the hash table. | ||||
|  * "the usual" patch-up process for the app-order doubly-linked-list. | ||||
|  * The use of _hd_hh_del below deserves special explanation. | ||||
|  * These used to be expressed using (delptr) but that led to a bug | ||||
|  * if someone used the same symbol for the head and deletee, like | ||||
|  *  HASH_DELETE(hh,users,users); | ||||
|  * We want that to work, but by changing the head (users) below | ||||
|  * we were forfeiting our ability to further refer to the deletee (users) | ||||
|  * in the patch-up process. Solution: use scratch space to | ||||
|  * copy the deletee pointer, then the latter references are via that | ||||
|  * scratch pointer rather than through the repointed (users) symbol. | ||||
|  */ | ||||
| #define HASH_DELETE(hh,head,delptr)                                              \ | ||||
| do {                                                                             \ | ||||
|     unsigned _hd_bkt;                                                            \ | ||||
|     struct UT_hash_handle *_hd_hh_del;                                           \ | ||||
|     if ( ((delptr)->hh.prev == NULL) && ((delptr)->hh.next == NULL) )  {         \ | ||||
|         uthash_free((head)->hh.tbl->buckets,                                     \ | ||||
|                     (head)->hh.tbl->num_buckets*sizeof(struct UT_hash_bucket) ); \ | ||||
|         HASH_BLOOM_FREE((head)->hh.tbl);                                         \ | ||||
|         uthash_free((head)->hh.tbl, sizeof(UT_hash_table));                      \ | ||||
|         head = NULL;                                                             \ | ||||
|     } else {                                                                     \ | ||||
|         _hd_hh_del = &((delptr)->hh);                                            \ | ||||
|         if ((delptr) == ELMT_FROM_HH((head)->hh.tbl,(head)->hh.tbl->tail)) {     \ | ||||
|             (head)->hh.tbl->tail =                                               \ | ||||
|                 (UT_hash_handle*)((ptrdiff_t)((delptr)->hh.prev) +               \ | ||||
|                 (head)->hh.tbl->hho);                                            \ | ||||
|         }                                                                        \ | ||||
|         if ((delptr)->hh.prev) {                                                 \ | ||||
|             ((UT_hash_handle*)((ptrdiff_t)((delptr)->hh.prev) +                  \ | ||||
|                     (head)->hh.tbl->hho))->next = (delptr)->hh.next;             \ | ||||
|         } else {                                                                 \ | ||||
|             DECLTYPE_ASSIGN(head,(delptr)->hh.next);                             \ | ||||
|         }                                                                        \ | ||||
|         if (_hd_hh_del->next) {                                                  \ | ||||
|             ((UT_hash_handle*)((ptrdiff_t)_hd_hh_del->next +                     \ | ||||
|                     (head)->hh.tbl->hho))->prev =                                \ | ||||
|                     _hd_hh_del->prev;                                            \ | ||||
|         }                                                                        \ | ||||
|         HASH_TO_BKT( _hd_hh_del->hashv, (head)->hh.tbl->num_buckets, _hd_bkt);   \ | ||||
|         HASH_DEL_IN_BKT(hh,(head)->hh.tbl->buckets[_hd_bkt], _hd_hh_del);        \ | ||||
|         (head)->hh.tbl->num_items--;                                             \ | ||||
|     }                                                                            \ | ||||
|     HASH_FSCK(hh,head);                                                          \ | ||||
| } while (0) | ||||
|  | ||||
|  | ||||
| /* convenience forms of HASH_FIND/HASH_ADD/HASH_DEL */ | ||||
| #define HASH_FIND_STR(head,findstr,out)                                          \ | ||||
|     HASH_FIND(hh,head,findstr,strlen(findstr),out) | ||||
| #define HASH_ADD_STR(head,strfield,add)                                          \ | ||||
|     HASH_ADD(hh,head,strfield[0],strlen(add->strfield),add) | ||||
| #define HASH_REPLACE_STR(head,strfield,add,replaced)                             \ | ||||
|     HASH_REPLACE(hh,head,strfield[0],strlen(add->strfield),add,replaced) | ||||
| #define HASH_FIND_INT(head,findint,out)                                          \ | ||||
|     HASH_FIND(hh,head,findint,sizeof(int),out) | ||||
| #define HASH_ADD_INT(head,intfield,add)                                          \ | ||||
|     HASH_ADD(hh,head,intfield,sizeof(int),add) | ||||
| #define HASH_REPLACE_INT(head,intfield,add,replaced)                             \ | ||||
|     HASH_REPLACE(hh,head,intfield,sizeof(int),add,replaced) | ||||
| #define HASH_FIND_PTR(head,findptr,out)                                          \ | ||||
|     HASH_FIND(hh,head,findptr,sizeof(void *),out) | ||||
| #define HASH_ADD_PTR(head,ptrfield,add)                                          \ | ||||
|     HASH_ADD(hh,head,ptrfield,sizeof(void *),add) | ||||
| #define HASH_REPLACE_PTR(head,ptrfield,add,replaced)                             \ | ||||
|     HASH_REPLACE(hh,head,ptrfield,sizeof(void *),add,replaced) | ||||
| #define HASH_DEL(head,delptr)                                                    \ | ||||
|     HASH_DELETE(hh,head,delptr) | ||||
|  | ||||
| /* HASH_FSCK checks hash integrity on every add/delete when HASH_DEBUG is defined. | ||||
|  * This is for uthash developer only; it compiles away if HASH_DEBUG isn't defined. | ||||
|  */ | ||||
| #ifdef HASH_DEBUG | ||||
| #define HASH_OOPS(...) do { fprintf(stderr,__VA_ARGS__); exit(-1); } while (0) | ||||
| #define HASH_FSCK(hh,head)                                                       \ | ||||
| do {                                                                             \ | ||||
|     unsigned _bkt_i;                                                             \ | ||||
|     unsigned _count, _bkt_count;                                                 \ | ||||
|     char *_prev;                                                                 \ | ||||
|     struct UT_hash_handle *_thh;                                                 \ | ||||
|     if (head) {                                                                  \ | ||||
|         _count = 0;                                                              \ | ||||
|         for( _bkt_i = 0; _bkt_i < (head)->hh.tbl->num_buckets; _bkt_i++) {       \ | ||||
|             _bkt_count = 0;                                                      \ | ||||
|             _thh = (head)->hh.tbl->buckets[_bkt_i].hh_head;                      \ | ||||
|             _prev = NULL;                                                        \ | ||||
|             while (_thh) {                                                       \ | ||||
|                if (_prev != (char*)(_thh->hh_prev)) {                            \ | ||||
|                    HASH_OOPS("invalid hh_prev %p, actual %p\n",                  \ | ||||
|                     _thh->hh_prev, _prev );                                      \ | ||||
|                }                                                                 \ | ||||
|                _bkt_count++;                                                     \ | ||||
|                _prev = (char*)(_thh);                                            \ | ||||
|                _thh = _thh->hh_next;                                             \ | ||||
|             }                                                                    \ | ||||
|             _count += _bkt_count;                                                \ | ||||
|             if ((head)->hh.tbl->buckets[_bkt_i].count !=  _bkt_count) {          \ | ||||
|                HASH_OOPS("invalid bucket count %d, actual %d\n",                 \ | ||||
|                 (head)->hh.tbl->buckets[_bkt_i].count, _bkt_count);              \ | ||||
|             }                                                                    \ | ||||
|         }                                                                        \ | ||||
|         if (_count != (head)->hh.tbl->num_items) {                               \ | ||||
|             HASH_OOPS("invalid hh item count %d, actual %d\n",                   \ | ||||
|                 (head)->hh.tbl->num_items, _count );                             \ | ||||
|         }                                                                        \ | ||||
|         /* traverse hh in app order; check next/prev integrity, count */         \ | ||||
|         _count = 0;                                                              \ | ||||
|         _prev = NULL;                                                            \ | ||||
|         _thh =  &(head)->hh;                                                     \ | ||||
|         while (_thh) {                                                           \ | ||||
|            _count++;                                                             \ | ||||
|            if (_prev !=(char*)(_thh->prev)) {                                    \ | ||||
|               HASH_OOPS("invalid prev %p, actual %p\n",                          \ | ||||
|                     _thh->prev, _prev );                                         \ | ||||
|            }                                                                     \ | ||||
|            _prev = (char*)ELMT_FROM_HH((head)->hh.tbl, _thh);                    \ | ||||
|            _thh = ( _thh->next ?  (UT_hash_handle*)((char*)(_thh->next) +        \ | ||||
|                                   (head)->hh.tbl->hho) : NULL );                 \ | ||||
|         }                                                                        \ | ||||
|         if (_count != (head)->hh.tbl->num_items) {                               \ | ||||
|             HASH_OOPS("invalid app item count %d, actual %d\n",                  \ | ||||
|                 (head)->hh.tbl->num_items, _count );                             \ | ||||
|         }                                                                        \ | ||||
|     }                                                                            \ | ||||
| } while (0) | ||||
| #else | ||||
| #define HASH_FSCK(hh,head)  | ||||
| #endif | ||||
|  | ||||
| /* When compiled with -DHASH_EMIT_KEYS, length-prefixed keys are emitted to  | ||||
|  * the descriptor to which this macro is defined for tuning the hash function. | ||||
|  * The app can #include <unistd.h> to get the prototype for write(2). */ | ||||
| #ifdef HASH_EMIT_KEYS | ||||
| #define HASH_EMIT_KEY(hh,head,keyptr,fieldlen)                                   \ | ||||
| do {                                                                             \ | ||||
|     unsigned _klen = fieldlen;                                                   \ | ||||
|     write(HASH_EMIT_KEYS, &_klen, sizeof(_klen));                                \ | ||||
|     write(HASH_EMIT_KEYS, keyptr, fieldlen);                                     \ | ||||
| } while (0) | ||||
| #else  | ||||
| #define HASH_EMIT_KEY(hh,head,keyptr,fieldlen)                     | ||||
| #endif | ||||
|  | ||||
| /* default to Jenkin's hash unless overridden e.g. DHASH_FUNCTION=HASH_SAX */ | ||||
| #ifdef HASH_FUNCTION  | ||||
| #define HASH_FCN HASH_FUNCTION | ||||
| #else | ||||
| #define HASH_FCN HASH_JEN | ||||
| #endif | ||||
|  | ||||
| /* The Bernstein hash function, used in Perl prior to v5.6. Note (x<<5+x)=x*33. */ | ||||
| #define HASH_BER(key,keylen,num_bkts,hashv,bkt)                                  \ | ||||
| do {                                                                             \ | ||||
|   unsigned _hb_keylen=keylen;                                                    \ | ||||
|   char *_hb_key=(char*)(key);                                                    \ | ||||
|   (hashv) = 0;                                                                   \ | ||||
|   while (_hb_keylen--)  { (hashv) = (((hashv) << 5) + (hashv)) + *_hb_key++; }   \ | ||||
|   bkt = (hashv) & (num_bkts-1);                                                  \ | ||||
| } while (0) | ||||
|  | ||||
|  | ||||
| /* SAX/FNV/OAT/JEN hash functions are macro variants of those listed at  | ||||
|  * http://eternallyconfuzzled.com/tuts/algorithms/jsw_tut_hashing.aspx */ | ||||
| #define HASH_SAX(key,keylen,num_bkts,hashv,bkt)                                  \ | ||||
| do {                                                                             \ | ||||
|   unsigned _sx_i;                                                                \ | ||||
|   char *_hs_key=(char*)(key);                                                    \ | ||||
|   hashv = 0;                                                                     \ | ||||
|   for(_sx_i=0; _sx_i < keylen; _sx_i++)                                          \ | ||||
|       hashv ^= (hashv << 5) + (hashv >> 2) + _hs_key[_sx_i];                     \ | ||||
|   bkt = hashv & (num_bkts-1);                                                    \ | ||||
| } while (0) | ||||
| /* FNV-1a variation */ | ||||
| #define HASH_FNV(key,keylen,num_bkts,hashv,bkt)                                  \ | ||||
| do {                                                                             \ | ||||
|   unsigned _fn_i;                                                                \ | ||||
|   char *_hf_key=(char*)(key);                                                    \ | ||||
|   hashv = 2166136261UL;                                                          \ | ||||
|   for(_fn_i=0; _fn_i < keylen; _fn_i++)                                          \ | ||||
|       hashv = hashv ^ _hf_key[_fn_i];                                            \ | ||||
|       hashv = hashv * 16777619;                                                  \ | ||||
|   bkt = hashv & (num_bkts-1);                                                    \ | ||||
| } while(0)  | ||||
|   | ||||
| #define HASH_OAT(key,keylen,num_bkts,hashv,bkt)                                  \ | ||||
| do {                                                                             \ | ||||
|   unsigned _ho_i;                                                                \ | ||||
|   char *_ho_key=(char*)(key);                                                    \ | ||||
|   hashv = 0;                                                                     \ | ||||
|   for(_ho_i=0; _ho_i < keylen; _ho_i++) {                                        \ | ||||
|       hashv += _ho_key[_ho_i];                                                   \ | ||||
|       hashv += (hashv << 10);                                                    \ | ||||
|       hashv ^= (hashv >> 6);                                                     \ | ||||
|   }                                                                              \ | ||||
|   hashv += (hashv << 3);                                                         \ | ||||
|   hashv ^= (hashv >> 11);                                                        \ | ||||
|   hashv += (hashv << 15);                                                        \ | ||||
|   bkt = hashv & (num_bkts-1);                                                    \ | ||||
| } while(0) | ||||
|  | ||||
| #define HASH_JEN_MIX(a,b,c)                                                      \ | ||||
| do {                                                                             \ | ||||
|   a -= b; a -= c; a ^= ( c >> 13 );                                              \ | ||||
|   b -= c; b -= a; b ^= ( a << 8 );                                               \ | ||||
|   c -= a; c -= b; c ^= ( b >> 13 );                                              \ | ||||
|   a -= b; a -= c; a ^= ( c >> 12 );                                              \ | ||||
|   b -= c; b -= a; b ^= ( a << 16 );                                              \ | ||||
|   c -= a; c -= b; c ^= ( b >> 5 );                                               \ | ||||
|   a -= b; a -= c; a ^= ( c >> 3 );                                               \ | ||||
|   b -= c; b -= a; b ^= ( a << 10 );                                              \ | ||||
|   c -= a; c -= b; c ^= ( b >> 15 );                                              \ | ||||
| } while (0) | ||||
|  | ||||
| #define HASH_JEN(key,keylen,num_bkts,hashv,bkt)                                  \ | ||||
| do {                                                                             \ | ||||
|   unsigned _hj_i,_hj_j,_hj_k;                                                    \ | ||||
|   unsigned char *_hj_key=(unsigned char*)(key);                                  \ | ||||
|   hashv = 0xfeedbeef;                                                            \ | ||||
|   _hj_i = _hj_j = 0x9e3779b9;                                                    \ | ||||
|   _hj_k = (unsigned)(keylen);                                                      \ | ||||
|   while (_hj_k >= 12) {                                                          \ | ||||
|     _hj_i +=    (_hj_key[0] + ( (unsigned)_hj_key[1] << 8 )                      \ | ||||
|         + ( (unsigned)_hj_key[2] << 16 )                                         \ | ||||
|         + ( (unsigned)_hj_key[3] << 24 ) );                                      \ | ||||
|     _hj_j +=    (_hj_key[4] + ( (unsigned)_hj_key[5] << 8 )                      \ | ||||
|         + ( (unsigned)_hj_key[6] << 16 )                                         \ | ||||
|         + ( (unsigned)_hj_key[7] << 24 ) );                                      \ | ||||
|     hashv += (_hj_key[8] + ( (unsigned)_hj_key[9] << 8 )                         \ | ||||
|         + ( (unsigned)_hj_key[10] << 16 )                                        \ | ||||
|         + ( (unsigned)_hj_key[11] << 24 ) );                                     \ | ||||
|                                                                                  \ | ||||
|      HASH_JEN_MIX(_hj_i, _hj_j, hashv);                                          \ | ||||
|                                                                                  \ | ||||
|      _hj_key += 12;                                                              \ | ||||
|      _hj_k -= 12;                                                                \ | ||||
|   }                                                                              \ | ||||
|   hashv += keylen;                                                               \ | ||||
|   switch ( _hj_k ) {                                                             \ | ||||
|      case 11: hashv += ( (unsigned)_hj_key[10] << 24 );                          \ | ||||
|      case 10: hashv += ( (unsigned)_hj_key[9] << 16 );                           \ | ||||
|      case 9:  hashv += ( (unsigned)_hj_key[8] << 8 );                            \ | ||||
|      case 8:  _hj_j += ( (unsigned)_hj_key[7] << 24 );                           \ | ||||
|      case 7:  _hj_j += ( (unsigned)_hj_key[6] << 16 );                           \ | ||||
|      case 6:  _hj_j += ( (unsigned)_hj_key[5] << 8 );                            \ | ||||
|      case 5:  _hj_j += _hj_key[4];                                               \ | ||||
|      case 4:  _hj_i += ( (unsigned)_hj_key[3] << 24 );                           \ | ||||
|      case 3:  _hj_i += ( (unsigned)_hj_key[2] << 16 );                           \ | ||||
|      case 2:  _hj_i += ( (unsigned)_hj_key[1] << 8 );                            \ | ||||
|      case 1:  _hj_i += _hj_key[0];                                               \ | ||||
|   }                                                                              \ | ||||
|   HASH_JEN_MIX(_hj_i, _hj_j, hashv);                                             \ | ||||
|   bkt = hashv & (num_bkts-1);                                                    \ | ||||
| } while(0) | ||||
|  | ||||
| /* The Paul Hsieh hash function */ | ||||
| #undef get16bits | ||||
| #if (defined(__GNUC__) && defined(__i386__)) || defined(__WATCOMC__)             \ | ||||
|   || defined(_MSC_VER) || defined (__BORLANDC__) || defined (__TURBOC__) | ||||
| #define get16bits(d) (*((const uint16_t *) (d))) | ||||
| #endif | ||||
|  | ||||
| #if !defined (get16bits) | ||||
| #define get16bits(d) ((((uint32_t)(((const uint8_t *)(d))[1])) << 8)             \ | ||||
|                        +(uint32_t)(((const uint8_t *)(d))[0]) ) | ||||
| #endif | ||||
| #define HASH_SFH(key,keylen,num_bkts,hashv,bkt)                                  \ | ||||
| do {                                                                             \ | ||||
|   unsigned char *_sfh_key=(unsigned char*)(key);                                 \ | ||||
|   uint32_t _sfh_tmp, _sfh_len = keylen;                                          \ | ||||
|                                                                                  \ | ||||
|   int _sfh_rem = _sfh_len & 3;                                                   \ | ||||
|   _sfh_len >>= 2;                                                                \ | ||||
|   hashv = 0xcafebabe;                                                            \ | ||||
|                                                                                  \ | ||||
|   /* Main loop */                                                                \ | ||||
|   for (;_sfh_len > 0; _sfh_len--) {                                              \ | ||||
|     hashv    += get16bits (_sfh_key);                                            \ | ||||
|     _sfh_tmp       = (uint32_t)(get16bits (_sfh_key+2)) << 11  ^ hashv;          \ | ||||
|     hashv     = (hashv << 16) ^ _sfh_tmp;                                        \ | ||||
|     _sfh_key += 2*sizeof (uint16_t);                                             \ | ||||
|     hashv    += hashv >> 11;                                                     \ | ||||
|   }                                                                              \ | ||||
|                                                                                  \ | ||||
|   /* Handle end cases */                                                         \ | ||||
|   switch (_sfh_rem) {                                                            \ | ||||
|     case 3: hashv += get16bits (_sfh_key);                                       \ | ||||
|             hashv ^= hashv << 16;                                                \ | ||||
|             hashv ^= (uint32_t)(_sfh_key[sizeof (uint16_t)] << 18);              \ | ||||
|             hashv += hashv >> 11;                                                \ | ||||
|             break;                                                               \ | ||||
|     case 2: hashv += get16bits (_sfh_key);                                       \ | ||||
|             hashv ^= hashv << 11;                                                \ | ||||
|             hashv += hashv >> 17;                                                \ | ||||
|             break;                                                               \ | ||||
|     case 1: hashv += *_sfh_key;                                                  \ | ||||
|             hashv ^= hashv << 10;                                                \ | ||||
|             hashv += hashv >> 1;                                                 \ | ||||
|   }                                                                              \ | ||||
|                                                                                  \ | ||||
|     /* Force "avalanching" of final 127 bits */                                  \ | ||||
|     hashv ^= hashv << 3;                                                         \ | ||||
|     hashv += hashv >> 5;                                                         \ | ||||
|     hashv ^= hashv << 4;                                                         \ | ||||
|     hashv += hashv >> 17;                                                        \ | ||||
|     hashv ^= hashv << 25;                                                        \ | ||||
|     hashv += hashv >> 6;                                                         \ | ||||
|     bkt = hashv & (num_bkts-1);                                                  \ | ||||
| } while(0)  | ||||
|  | ||||
| #ifdef HASH_USING_NO_STRICT_ALIASING | ||||
| /* The MurmurHash exploits some CPU's (x86,x86_64) tolerance for unaligned reads. | ||||
|  * For other types of CPU's (e.g. Sparc) an unaligned read causes a bus error. | ||||
|  * MurmurHash uses the faster approach only on CPU's where we know it's safe.  | ||||
|  * | ||||
|  * Note the preprocessor built-in defines can be emitted using: | ||||
|  * | ||||
|  *   gcc -m64 -dM -E - < /dev/null                  (on gcc) | ||||
|  *   cc -## a.c (where a.c is a simple test file)   (Sun Studio) | ||||
|  */ | ||||
| #if (defined(__i386__) || defined(__x86_64__)  || defined(_M_IX86)) | ||||
| #define MUR_GETBLOCK(p,i) p[i] | ||||
| #else /* non intel */ | ||||
| #define MUR_PLUS0_ALIGNED(p) (((unsigned long)p & 0x3) == 0) | ||||
| #define MUR_PLUS1_ALIGNED(p) (((unsigned long)p & 0x3) == 1) | ||||
| #define MUR_PLUS2_ALIGNED(p) (((unsigned long)p & 0x3) == 2) | ||||
| #define MUR_PLUS3_ALIGNED(p) (((unsigned long)p & 0x3) == 3) | ||||
| #define WP(p) ((uint32_t*)((unsigned long)(p) & ~3UL)) | ||||
| #if (defined(__BIG_ENDIAN__) || defined(SPARC) || defined(__ppc__) || defined(__ppc64__)) | ||||
| #define MUR_THREE_ONE(p) ((((*WP(p))&0x00ffffff) << 8) | (((*(WP(p)+1))&0xff000000) >> 24)) | ||||
| #define MUR_TWO_TWO(p)   ((((*WP(p))&0x0000ffff) <<16) | (((*(WP(p)+1))&0xffff0000) >> 16)) | ||||
| #define MUR_ONE_THREE(p) ((((*WP(p))&0x000000ff) <<24) | (((*(WP(p)+1))&0xffffff00) >>  8)) | ||||
| #else /* assume little endian non-intel */ | ||||
| #define MUR_THREE_ONE(p) ((((*WP(p))&0xffffff00) >> 8) | (((*(WP(p)+1))&0x000000ff) << 24)) | ||||
| #define MUR_TWO_TWO(p)   ((((*WP(p))&0xffff0000) >>16) | (((*(WP(p)+1))&0x0000ffff) << 16)) | ||||
| #define MUR_ONE_THREE(p) ((((*WP(p))&0xff000000) >>24) | (((*(WP(p)+1))&0x00ffffff) <<  8)) | ||||
| #endif | ||||
| #define MUR_GETBLOCK(p,i) (MUR_PLUS0_ALIGNED(p) ? ((p)[i]) :           \ | ||||
|                             (MUR_PLUS1_ALIGNED(p) ? MUR_THREE_ONE(p) : \ | ||||
|                              (MUR_PLUS2_ALIGNED(p) ? MUR_TWO_TWO(p) :  \ | ||||
|                                                       MUR_ONE_THREE(p)))) | ||||
| #endif | ||||
| #define MUR_ROTL32(x,r) (((x) << (r)) | ((x) >> (32 - (r)))) | ||||
| #define MUR_FMIX(_h) \ | ||||
| do {                 \ | ||||
|   _h ^= _h >> 16;    \ | ||||
|   _h *= 0x85ebca6b;  \ | ||||
|   _h ^= _h >> 13;    \ | ||||
|   _h *= 0xc2b2ae35l; \ | ||||
|   _h ^= _h >> 16;    \ | ||||
| } while(0) | ||||
|  | ||||
| #define HASH_MUR(key,keylen,num_bkts,hashv,bkt)                        \ | ||||
| do {                                                                   \ | ||||
|   const uint8_t *_mur_data = (const uint8_t*)(key);                    \ | ||||
|   const int _mur_nblocks = (keylen) / 4;                               \ | ||||
|   uint32_t _mur_h1 = 0xf88D5353;                                       \ | ||||
|   uint32_t _mur_c1 = 0xcc9e2d51;                                       \ | ||||
|   uint32_t _mur_c2 = 0x1b873593;                                       \ | ||||
|   uint32_t _mur_k1 = 0;                                                \ | ||||
|   const uint8_t *_mur_tail;                                            \ | ||||
|   const uint32_t *_mur_blocks = (const uint32_t*)(_mur_data+_mur_nblocks*4); \ | ||||
|   int _mur_i;                                                          \ | ||||
|   for(_mur_i = -_mur_nblocks; _mur_i; _mur_i++) {                      \ | ||||
|     _mur_k1 = MUR_GETBLOCK(_mur_blocks,_mur_i);                        \ | ||||
|     _mur_k1 *= _mur_c1;                                                \ | ||||
|     _mur_k1 = MUR_ROTL32(_mur_k1,15);                                  \ | ||||
|     _mur_k1 *= _mur_c2;                                                \ | ||||
|                                                                        \ | ||||
|     _mur_h1 ^= _mur_k1;                                                \ | ||||
|     _mur_h1 = MUR_ROTL32(_mur_h1,13);                                  \ | ||||
|     _mur_h1 = _mur_h1*5+0xe6546b64;                                    \ | ||||
|   }                                                                    \ | ||||
|   _mur_tail = (const uint8_t*)(_mur_data + _mur_nblocks*4);            \ | ||||
|   _mur_k1=0;                                                           \ | ||||
|   switch((keylen) & 3) {                                               \ | ||||
|     case 3: _mur_k1 ^= _mur_tail[2] << 16;                             \ | ||||
|     case 2: _mur_k1 ^= _mur_tail[1] << 8;                              \ | ||||
|     case 1: _mur_k1 ^= _mur_tail[0];                                   \ | ||||
|     _mur_k1 *= _mur_c1;                                                \ | ||||
|     _mur_k1 = MUR_ROTL32(_mur_k1,15);                                  \ | ||||
|     _mur_k1 *= _mur_c2;                                                \ | ||||
|     _mur_h1 ^= _mur_k1;                                                \ | ||||
|   }                                                                    \ | ||||
|   _mur_h1 ^= (keylen);                                                 \ | ||||
|   MUR_FMIX(_mur_h1);                                                   \ | ||||
|   hashv = _mur_h1;                                                     \ | ||||
|   bkt = hashv & (num_bkts-1);                                          \ | ||||
| } while(0) | ||||
| #endif  /* HASH_USING_NO_STRICT_ALIASING */ | ||||
|  | ||||
| /* key comparison function; return 0 if keys equal */ | ||||
| #define HASH_KEYCMP(a,b,len) memcmp(a,b,len)  | ||||
|  | ||||
| /* iterate over items in a known bucket to find desired item */ | ||||
| #define HASH_FIND_IN_BKT(tbl,hh,head,keyptr,keylen_in,out)                       \ | ||||
| do {                                                                             \ | ||||
|  if (head.hh_head) DECLTYPE_ASSIGN(out,ELMT_FROM_HH(tbl,head.hh_head));          \ | ||||
|  else out=NULL;                                                                  \ | ||||
|  while (out) {                                                                   \ | ||||
|     if ((out)->hh.keylen == keylen_in) {                                           \ | ||||
|         if ((HASH_KEYCMP((out)->hh.key,keyptr,keylen_in)) == 0) break;             \ | ||||
|     }                                                                            \ | ||||
|     if ((out)->hh.hh_next) DECLTYPE_ASSIGN(out,ELMT_FROM_HH(tbl,(out)->hh.hh_next)); \ | ||||
|     else out = NULL;                                                             \ | ||||
|  }                                                                               \ | ||||
| } while(0) | ||||
|  | ||||
| /* add an item to a bucket  */ | ||||
| #define HASH_ADD_TO_BKT(head,addhh)                                              \ | ||||
| do {                                                                             \ | ||||
|  head.count++;                                                                   \ | ||||
|  (addhh)->hh_next = head.hh_head;                                                \ | ||||
|  (addhh)->hh_prev = NULL;                                                        \ | ||||
|  if (head.hh_head) { (head).hh_head->hh_prev = (addhh); }                        \ | ||||
|  (head).hh_head=addhh;                                                           \ | ||||
|  if (head.count >= ((head.expand_mult+1) * HASH_BKT_CAPACITY_THRESH)             \ | ||||
|      && (addhh)->tbl->noexpand != 1) {                                           \ | ||||
|        HASH_EXPAND_BUCKETS((addhh)->tbl);                                        \ | ||||
|  }                                                                               \ | ||||
| } while(0) | ||||
|  | ||||
| /* remove an item from a given bucket */ | ||||
| #define HASH_DEL_IN_BKT(hh,head,hh_del)                                          \ | ||||
|     (head).count--;                                                              \ | ||||
|     if ((head).hh_head == hh_del) {                                              \ | ||||
|       (head).hh_head = hh_del->hh_next;                                          \ | ||||
|     }                                                                            \ | ||||
|     if (hh_del->hh_prev) {                                                       \ | ||||
|         hh_del->hh_prev->hh_next = hh_del->hh_next;                              \ | ||||
|     }                                                                            \ | ||||
|     if (hh_del->hh_next) {                                                       \ | ||||
|         hh_del->hh_next->hh_prev = hh_del->hh_prev;                              \ | ||||
|     }                                                                 | ||||
|  | ||||
| /* Bucket expansion has the effect of doubling the number of buckets | ||||
|  * and redistributing the items into the new buckets. Ideally the | ||||
|  * items will distribute more or less evenly into the new buckets | ||||
|  * (the extent to which this is true is a measure of the quality of | ||||
|  * the hash function as it applies to the key domain).  | ||||
|  *  | ||||
|  * With the items distributed into more buckets, the chain length | ||||
|  * (item count) in each bucket is reduced. Thus by expanding buckets | ||||
|  * the hash keeps a bound on the chain length. This bounded chain  | ||||
|  * length is the essence of how a hash provides constant time lookup. | ||||
|  *  | ||||
|  * The calculation of tbl->ideal_chain_maxlen below deserves some | ||||
|  * explanation. First, keep in mind that we're calculating the ideal | ||||
|  * maximum chain length based on the *new* (doubled) bucket count. | ||||
|  * In fractions this is just n/b (n=number of items,b=new num buckets). | ||||
|  * Since the ideal chain length is an integer, we want to calculate  | ||||
|  * ceil(n/b). We don't depend on floating point arithmetic in this | ||||
|  * hash, so to calculate ceil(n/b) with integers we could write | ||||
|  *  | ||||
|  *      ceil(n/b) = (n/b) + ((n%b)?1:0) | ||||
|  *  | ||||
|  * and in fact a previous version of this hash did just that. | ||||
|  * But now we have improved things a bit by recognizing that b is | ||||
|  * always a power of two. We keep its base 2 log handy (call it lb), | ||||
|  * so now we can write this with a bit shift and logical AND: | ||||
|  *  | ||||
|  *      ceil(n/b) = (n>>lb) + ( (n & (b-1)) ? 1:0) | ||||
|  *  | ||||
|  */ | ||||
| #define HASH_EXPAND_BUCKETS(tbl)                                                 \ | ||||
| do {                                                                             \ | ||||
|     unsigned _he_bkt;                                                            \ | ||||
|     unsigned _he_bkt_i;                                                          \ | ||||
|     struct UT_hash_handle *_he_thh, *_he_hh_nxt;                                 \ | ||||
|     UT_hash_bucket *_he_new_buckets, *_he_newbkt;                                \ | ||||
|     _he_new_buckets = (UT_hash_bucket*)uthash_malloc(                            \ | ||||
|              2 * tbl->num_buckets * sizeof(struct UT_hash_bucket));              \ | ||||
|     if (!_he_new_buckets) { uthash_fatal( "out of memory"); }                    \ | ||||
|     memset(_he_new_buckets, 0,                                                   \ | ||||
|             2 * tbl->num_buckets * sizeof(struct UT_hash_bucket));               \ | ||||
|     tbl->ideal_chain_maxlen =                                                    \ | ||||
|        (tbl->num_items >> (tbl->log2_num_buckets+1)) +                           \ | ||||
|        ((tbl->num_items & ((tbl->num_buckets*2)-1)) ? 1 : 0);                    \ | ||||
|     tbl->nonideal_items = 0;                                                     \ | ||||
|     for(_he_bkt_i = 0; _he_bkt_i < tbl->num_buckets; _he_bkt_i++)                \ | ||||
|     {                                                                            \ | ||||
|         _he_thh = tbl->buckets[ _he_bkt_i ].hh_head;                             \ | ||||
|         while (_he_thh) {                                                        \ | ||||
|            _he_hh_nxt = _he_thh->hh_next;                                        \ | ||||
|            HASH_TO_BKT( _he_thh->hashv, tbl->num_buckets*2, _he_bkt);            \ | ||||
|            _he_newbkt = &(_he_new_buckets[ _he_bkt ]);                           \ | ||||
|            if (++(_he_newbkt->count) > tbl->ideal_chain_maxlen) {                \ | ||||
|              tbl->nonideal_items++;                                              \ | ||||
|              _he_newbkt->expand_mult = _he_newbkt->count /                       \ | ||||
|                                         tbl->ideal_chain_maxlen;                 \ | ||||
|            }                                                                     \ | ||||
|            _he_thh->hh_prev = NULL;                                              \ | ||||
|            _he_thh->hh_next = _he_newbkt->hh_head;                               \ | ||||
|            if (_he_newbkt->hh_head) _he_newbkt->hh_head->hh_prev =               \ | ||||
|                 _he_thh;                                                         \ | ||||
|            _he_newbkt->hh_head = _he_thh;                                        \ | ||||
|            _he_thh = _he_hh_nxt;                                                 \ | ||||
|         }                                                                        \ | ||||
|     }                                                                            \ | ||||
|     uthash_free( tbl->buckets, tbl->num_buckets*sizeof(struct UT_hash_bucket) ); \ | ||||
|     tbl->num_buckets *= 2;                                                       \ | ||||
|     tbl->log2_num_buckets++;                                                     \ | ||||
|     tbl->buckets = _he_new_buckets;                                              \ | ||||
|     tbl->ineff_expands = (tbl->nonideal_items > (tbl->num_items >> 1)) ?         \ | ||||
|         (tbl->ineff_expands+1) : 0;                                              \ | ||||
|     if (tbl->ineff_expands > 1) {                                                \ | ||||
|         tbl->noexpand=1;                                                         \ | ||||
|         uthash_noexpand_fyi(tbl);                                                \ | ||||
|     }                                                                            \ | ||||
|     uthash_expand_fyi(tbl);                                                      \ | ||||
| } while(0) | ||||
|  | ||||
|  | ||||
| /* This is an adaptation of Simon Tatham's O(n log(n)) mergesort */ | ||||
| /* Note that HASH_SORT assumes the hash handle name to be hh.  | ||||
|  * HASH_SRT was added to allow the hash handle name to be passed in. */ | ||||
| #define HASH_SORT(head,cmpfcn) HASH_SRT(hh,head,cmpfcn) | ||||
| #define HASH_SRT(hh,head,cmpfcn)                                                 \ | ||||
| do {                                                                             \ | ||||
|   unsigned _hs_i;                                                                \ | ||||
|   unsigned _hs_looping,_hs_nmerges,_hs_insize,_hs_psize,_hs_qsize;               \ | ||||
|   struct UT_hash_handle *_hs_p, *_hs_q, *_hs_e, *_hs_list, *_hs_tail;            \ | ||||
|   if (head) {                                                                    \ | ||||
|       _hs_insize = 1;                                                            \ | ||||
|       _hs_looping = 1;                                                           \ | ||||
|       _hs_list = &((head)->hh);                                                  \ | ||||
|       while (_hs_looping) {                                                      \ | ||||
|           _hs_p = _hs_list;                                                      \ | ||||
|           _hs_list = NULL;                                                       \ | ||||
|           _hs_tail = NULL;                                                       \ | ||||
|           _hs_nmerges = 0;                                                       \ | ||||
|           while (_hs_p) {                                                        \ | ||||
|               _hs_nmerges++;                                                     \ | ||||
|               _hs_q = _hs_p;                                                     \ | ||||
|               _hs_psize = 0;                                                     \ | ||||
|               for ( _hs_i = 0; _hs_i  < _hs_insize; _hs_i++ ) {                  \ | ||||
|                   _hs_psize++;                                                   \ | ||||
|                   _hs_q = (UT_hash_handle*)((_hs_q->next) ?                      \ | ||||
|                           ((void*)((char*)(_hs_q->next) +                        \ | ||||
|                           (head)->hh.tbl->hho)) : NULL);                         \ | ||||
|                   if (! (_hs_q) ) break;                                         \ | ||||
|               }                                                                  \ | ||||
|               _hs_qsize = _hs_insize;                                            \ | ||||
|               while ((_hs_psize > 0) || ((_hs_qsize > 0) && _hs_q )) {           \ | ||||
|                   if (_hs_psize == 0) {                                          \ | ||||
|                       _hs_e = _hs_q;                                             \ | ||||
|                       _hs_q = (UT_hash_handle*)((_hs_q->next) ?                  \ | ||||
|                               ((void*)((char*)(_hs_q->next) +                    \ | ||||
|                               (head)->hh.tbl->hho)) : NULL);                     \ | ||||
|                       _hs_qsize--;                                               \ | ||||
|                   } else if ( (_hs_qsize == 0) || !(_hs_q) ) {                   \ | ||||
|                       _hs_e = _hs_p;                                             \ | ||||
|                       if (_hs_p){                                                \ | ||||
|                         _hs_p = (UT_hash_handle*)((_hs_p->next) ?                \ | ||||
|                                 ((void*)((char*)(_hs_p->next) +                  \ | ||||
|                                 (head)->hh.tbl->hho)) : NULL);                   \ | ||||
|                        }                                                         \ | ||||
|                       _hs_psize--;                                               \ | ||||
|                   } else if ((                                                   \ | ||||
|                       cmpfcn(DECLTYPE(head)(ELMT_FROM_HH((head)->hh.tbl,_hs_p)), \ | ||||
|                              DECLTYPE(head)(ELMT_FROM_HH((head)->hh.tbl,_hs_q))) \ | ||||
|                              ) <= 0) {                                           \ | ||||
|                       _hs_e = _hs_p;                                             \ | ||||
|                       if (_hs_p){                                                \ | ||||
|                         _hs_p = (UT_hash_handle*)((_hs_p->next) ?                \ | ||||
|                                ((void*)((char*)(_hs_p->next) +                   \ | ||||
|                                (head)->hh.tbl->hho)) : NULL);                    \ | ||||
|                        }                                                         \ | ||||
|                       _hs_psize--;                                               \ | ||||
|                   } else {                                                       \ | ||||
|                       _hs_e = _hs_q;                                             \ | ||||
|                       _hs_q = (UT_hash_handle*)((_hs_q->next) ?                  \ | ||||
|                               ((void*)((char*)(_hs_q->next) +                    \ | ||||
|                               (head)->hh.tbl->hho)) : NULL);                     \ | ||||
|                       _hs_qsize--;                                               \ | ||||
|                   }                                                              \ | ||||
|                   if ( _hs_tail ) {                                              \ | ||||
|                       _hs_tail->next = ((_hs_e) ?                                \ | ||||
|                             ELMT_FROM_HH((head)->hh.tbl,_hs_e) : NULL);          \ | ||||
|                   } else {                                                       \ | ||||
|                       _hs_list = _hs_e;                                          \ | ||||
|                   }                                                              \ | ||||
|                   if (_hs_e) {                                                   \ | ||||
|                   _hs_e->prev = ((_hs_tail) ?                                    \ | ||||
|                      ELMT_FROM_HH((head)->hh.tbl,_hs_tail) : NULL);              \ | ||||
|                   }                                                              \ | ||||
|                   _hs_tail = _hs_e;                                              \ | ||||
|               }                                                                  \ | ||||
|               _hs_p = _hs_q;                                                     \ | ||||
|           }                                                                      \ | ||||
|           if (_hs_tail){                                                         \ | ||||
|             _hs_tail->next = NULL;                                               \ | ||||
|           }                                                                      \ | ||||
|           if ( _hs_nmerges <= 1 ) {                                              \ | ||||
|               _hs_looping=0;                                                     \ | ||||
|               (head)->hh.tbl->tail = _hs_tail;                                   \ | ||||
|               DECLTYPE_ASSIGN(head,ELMT_FROM_HH((head)->hh.tbl, _hs_list));      \ | ||||
|           }                                                                      \ | ||||
|           _hs_insize *= 2;                                                       \ | ||||
|       }                                                                          \ | ||||
|       HASH_FSCK(hh,head);                                                        \ | ||||
|  }                                                                               \ | ||||
| } while (0) | ||||
|  | ||||
| /* This function selects items from one hash into another hash.  | ||||
|  * The end result is that the selected items have dual presence  | ||||
|  * in both hashes. There is no copy of the items made; rather  | ||||
|  * they are added into the new hash through a secondary hash  | ||||
|  * hash handle that must be present in the structure. */ | ||||
| #define HASH_SELECT(hh_dst, dst, hh_src, src, cond)                              \ | ||||
| do {                                                                             \ | ||||
|   unsigned _src_bkt, _dst_bkt;                                                   \ | ||||
|   void *_last_elt=NULL, *_elt;                                                   \ | ||||
|   UT_hash_handle *_src_hh, *_dst_hh, *_last_elt_hh=NULL;                         \ | ||||
|   ptrdiff_t _dst_hho = ((char*)(&(dst)->hh_dst) - (char*)(dst));                 \ | ||||
|   if (src) {                                                                     \ | ||||
|     for(_src_bkt=0; _src_bkt < (src)->hh_src.tbl->num_buckets; _src_bkt++) {     \ | ||||
|       for(_src_hh = (src)->hh_src.tbl->buckets[_src_bkt].hh_head;                \ | ||||
|           _src_hh;                                                               \ | ||||
|           _src_hh = _src_hh->hh_next) {                                          \ | ||||
|           _elt = ELMT_FROM_HH((src)->hh_src.tbl, _src_hh);                       \ | ||||
|           if (cond(_elt)) {                                                      \ | ||||
|             _dst_hh = (UT_hash_handle*)(((char*)_elt) + _dst_hho);               \ | ||||
|             _dst_hh->key = _src_hh->key;                                         \ | ||||
|             _dst_hh->keylen = _src_hh->keylen;                                   \ | ||||
|             _dst_hh->hashv = _src_hh->hashv;                                     \ | ||||
|             _dst_hh->prev = _last_elt;                                           \ | ||||
|             _dst_hh->next = NULL;                                                \ | ||||
|             if (_last_elt_hh) { _last_elt_hh->next = _elt; }                     \ | ||||
|             if (!dst) {                                                          \ | ||||
|               DECLTYPE_ASSIGN(dst,_elt);                                         \ | ||||
|               HASH_MAKE_TABLE(hh_dst,dst);                                       \ | ||||
|             } else {                                                             \ | ||||
|               _dst_hh->tbl = (dst)->hh_dst.tbl;                                  \ | ||||
|             }                                                                    \ | ||||
|             HASH_TO_BKT(_dst_hh->hashv, _dst_hh->tbl->num_buckets, _dst_bkt);    \ | ||||
|             HASH_ADD_TO_BKT(_dst_hh->tbl->buckets[_dst_bkt],_dst_hh);            \ | ||||
|             (dst)->hh_dst.tbl->num_items++;                                      \ | ||||
|             _last_elt = _elt;                                                    \ | ||||
|             _last_elt_hh = _dst_hh;                                              \ | ||||
|           }                                                                      \ | ||||
|       }                                                                          \ | ||||
|     }                                                                            \ | ||||
|   }                                                                              \ | ||||
|   HASH_FSCK(hh_dst,dst);                                                         \ | ||||
| } while (0) | ||||
|  | ||||
| #define HASH_CLEAR(hh,head)                                                      \ | ||||
| do {                                                                             \ | ||||
|   if (head) {                                                                    \ | ||||
|     uthash_free((head)->hh.tbl->buckets,                                         \ | ||||
|                 (head)->hh.tbl->num_buckets*sizeof(struct UT_hash_bucket));      \ | ||||
|     HASH_BLOOM_FREE((head)->hh.tbl);                                             \ | ||||
|     uthash_free((head)->hh.tbl, sizeof(UT_hash_table));                          \ | ||||
|     (head)=NULL;                                                                 \ | ||||
|   }                                                                              \ | ||||
| } while(0) | ||||
|  | ||||
| #define HASH_OVERHEAD(hh,head)                                                   \ | ||||
|  (size_t)((((head)->hh.tbl->num_items   * sizeof(UT_hash_handle))   +            \ | ||||
|            ((head)->hh.tbl->num_buckets * sizeof(UT_hash_bucket))   +            \ | ||||
|             (sizeof(UT_hash_table))                                 +            \ | ||||
|             (HASH_BLOOM_BYTELEN))) | ||||
|  | ||||
| #ifdef NO_DECLTYPE | ||||
| #define HASH_ITER(hh,head,el,tmp)                                                \ | ||||
| for((el)=(head), (*(char**)(&(tmp)))=(char*)((head)?(head)->hh.next:NULL);       \ | ||||
|   el; (el)=(tmp),(*(char**)(&(tmp)))=(char*)((tmp)?(tmp)->hh.next:NULL))  | ||||
| #else | ||||
| #define HASH_ITER(hh,head,el,tmp)                                                \ | ||||
| for((el)=(head),(tmp)=DECLTYPE(el)((head)?(head)->hh.next:NULL);                 \ | ||||
|   el; (el)=(tmp),(tmp)=DECLTYPE(el)((tmp)?(tmp)->hh.next:NULL)) | ||||
| #endif | ||||
|  | ||||
| /* obtain a count of items in the hash */ | ||||
| #define HASH_COUNT(head) HASH_CNT(hh,head)  | ||||
| #define HASH_CNT(hh,head) ((head)?((head)->hh.tbl->num_items):0) | ||||
|  | ||||
| typedef struct UT_hash_bucket { | ||||
|    struct UT_hash_handle *hh_head; | ||||
|    unsigned count; | ||||
|  | ||||
|    /* expand_mult is normally set to 0. In this situation, the max chain length | ||||
|     * threshold is enforced at its default value, HASH_BKT_CAPACITY_THRESH. (If | ||||
|     * the bucket's chain exceeds this length, bucket expansion is triggered).  | ||||
|     * However, setting expand_mult to a non-zero value delays bucket expansion | ||||
|     * (that would be triggered by additions to this particular bucket) | ||||
|     * until its chain length reaches a *multiple* of HASH_BKT_CAPACITY_THRESH. | ||||
|     * (The multiplier is simply expand_mult+1). The whole idea of this | ||||
|     * multiplier is to reduce bucket expansions, since they are expensive, in | ||||
|     * situations where we know that a particular bucket tends to be overused. | ||||
|     * It is better to let its chain length grow to a longer yet-still-bounded | ||||
|     * value, than to do an O(n) bucket expansion too often.  | ||||
|     */ | ||||
|    unsigned expand_mult; | ||||
|  | ||||
| } UT_hash_bucket; | ||||
|  | ||||
| /* random signature used only to find hash tables in external analysis */ | ||||
| #define HASH_SIGNATURE 0xa0111fe1 | ||||
| #define HASH_BLOOM_SIGNATURE 0xb12220f2 | ||||
|  | ||||
| typedef struct UT_hash_table { | ||||
|    UT_hash_bucket *buckets; | ||||
|    unsigned num_buckets, log2_num_buckets; | ||||
|    unsigned num_items; | ||||
|    struct UT_hash_handle *tail; /* tail hh in app order, for fast append    */ | ||||
|    ptrdiff_t hho; /* hash handle offset (byte pos of hash handle in element */ | ||||
|  | ||||
|    /* in an ideal situation (all buckets used equally), no bucket would have | ||||
|     * more than ceil(#items/#buckets) items. that's the ideal chain length. */ | ||||
|    unsigned ideal_chain_maxlen; | ||||
|  | ||||
|    /* nonideal_items is the number of items in the hash whose chain position | ||||
|     * exceeds the ideal chain maxlen. these items pay the penalty for an uneven | ||||
|     * hash distribution; reaching them in a chain traversal takes >ideal steps */ | ||||
|    unsigned nonideal_items; | ||||
|  | ||||
|    /* ineffective expands occur when a bucket doubling was performed, but  | ||||
|     * afterward, more than half the items in the hash had nonideal chain | ||||
|     * positions. If this happens on two consecutive expansions we inhibit any | ||||
|     * further expansion, as it's not helping; this happens when the hash | ||||
|     * function isn't a good fit for the key domain. When expansion is inhibited | ||||
|     * the hash will still work, albeit no longer in constant time. */ | ||||
|    unsigned ineff_expands, noexpand; | ||||
|  | ||||
|    uint32_t signature; /* used only to find hash tables in external analysis */ | ||||
| #ifdef HASH_BLOOM | ||||
|    uint32_t bloom_sig; /* used only to test bloom exists in external analysis */ | ||||
|    uint8_t *bloom_bv; | ||||
|    char bloom_nbits; | ||||
| #endif | ||||
|  | ||||
| } UT_hash_table; | ||||
|  | ||||
| typedef struct UT_hash_handle { | ||||
|    struct UT_hash_table *tbl; | ||||
|    void *prev;                       /* prev element in app order      */ | ||||
|    void *next;                       /* next element in app order      */ | ||||
|    struct UT_hash_handle *hh_prev;   /* previous hh in bucket order    */ | ||||
|    struct UT_hash_handle *hh_next;   /* next hh in bucket order        */ | ||||
|    void *key;                        /* ptr to enclosing struct's key  */ | ||||
|    unsigned keylen;                  /* enclosing struct's key len     */ | ||||
|    unsigned hashv;                   /* result of hash-fcn(key)        */ | ||||
| } UT_hash_handle; | ||||
|  | ||||
| #endif /* UTHASH_H */ | ||||
							
								
								
									
										757
									
								
								src/ut/utlist.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										757
									
								
								src/ut/utlist.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,757 @@ | ||||
| /* | ||||
| Copyright (c) 2007-2014, Troy D. Hanson   http://troydhanson.github.com/uthash/ | ||||
| All rights reserved. | ||||
|  | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions are met: | ||||
|  | ||||
|     * Redistributions of source code must retain the above copyright | ||||
|       notice, this list of conditions and the following disclaimer. | ||||
|  | ||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS | ||||
| IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED | ||||
| TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A | ||||
| PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER | ||||
| OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | ||||
| EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | ||||
| PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | ||||
| PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF | ||||
| LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | ||||
| NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||||
| SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
| */ | ||||
|  | ||||
| #ifndef UTLIST_H | ||||
| #define UTLIST_H | ||||
|  | ||||
| #define UTLIST_VERSION 1.9.9 | ||||
|  | ||||
| #include <assert.h> | ||||
|  | ||||
| /*  | ||||
|  * This file contains macros to manipulate singly and doubly-linked lists. | ||||
|  * | ||||
|  * 1. LL_ macros:  singly-linked lists. | ||||
|  * 2. DL_ macros:  doubly-linked lists. | ||||
|  * 3. CDL_ macros: circular doubly-linked lists. | ||||
|  * | ||||
|  * To use singly-linked lists, your structure must have a "next" pointer. | ||||
|  * To use doubly-linked lists, your structure must "prev" and "next" pointers. | ||||
|  * Either way, the pointer to the head of the list must be initialized to NULL. | ||||
|  *  | ||||
|  * ----------------.EXAMPLE ------------------------- | ||||
|  * struct item { | ||||
|  *      int id; | ||||
|  *      struct item *prev, *next; | ||||
|  * } | ||||
|  * | ||||
|  * struct item *list = NULL: | ||||
|  * | ||||
|  * int main() { | ||||
|  *      struct item *item; | ||||
|  *      ... allocate and populate item ... | ||||
|  *      DL_APPEND(list, item); | ||||
|  * } | ||||
|  * -------------------------------------------------- | ||||
|  * | ||||
|  * For doubly-linked lists, the append and delete macros are O(1) | ||||
|  * For singly-linked lists, append and delete are O(n) but prepend is O(1) | ||||
|  * The sort macro is O(n log(n)) for all types of single/double/circular lists. | ||||
|  */ | ||||
|  | ||||
| /* These macros use decltype or the earlier __typeof GNU extension. | ||||
|    As decltype is only available in newer compilers (VS2010 or gcc 4.3+ | ||||
|    when compiling c++ code), this code uses whatever method is needed | ||||
|    or, for VS2008 where neither is available, uses casting workarounds. */ | ||||
| #ifdef _MSC_VER            /* MS compiler */ | ||||
| #if _MSC_VER >= 1600 && defined(__cplusplus)  /* VS2010 or newer in C++ mode */ | ||||
| #define LDECLTYPE(x) decltype(x) | ||||
| #else                     /* VS2008 or older (or VS2010 in C mode) */ | ||||
| #define NO_DECLTYPE | ||||
| #define LDECLTYPE(x) char* | ||||
| #endif | ||||
| #elif defined(__ICCARM__) | ||||
| #define NO_DECLTYPE | ||||
| #define LDECLTYPE(x) char* | ||||
| #else                      /* GNU, Sun and other compilers */ | ||||
| #define LDECLTYPE(x) __typeof(x) | ||||
| #endif | ||||
|  | ||||
| /* for VS2008 we use some workarounds to get around the lack of decltype, | ||||
|  * namely, we always reassign our tmp variable to the list head if we need | ||||
|  * to dereference its prev/next pointers, and save/restore the real head.*/ | ||||
| #ifdef NO_DECLTYPE | ||||
| #define _SV(elt,list) _tmp = (char*)(list); {char **_alias = (char**)&(list); *_alias = (elt); } | ||||
| #define _NEXT(elt,list,next) ((char*)((list)->next)) | ||||
| #define _NEXTASGN(elt,list,to,next) { char **_alias = (char**)&((list)->next); *_alias=(char*)(to); } | ||||
| /* #define _PREV(elt,list,prev) ((char*)((list)->prev)) */ | ||||
| #define _PREVASGN(elt,list,to,prev) { char **_alias = (char**)&((list)->prev); *_alias=(char*)(to); } | ||||
| #define _RS(list) { char **_alias = (char**)&(list); *_alias=_tmp; } | ||||
| #define _CASTASGN(a,b) { char **_alias = (char**)&(a); *_alias=(char*)(b); } | ||||
| #else  | ||||
| #define _SV(elt,list) | ||||
| #define _NEXT(elt,list,next) ((elt)->next) | ||||
| #define _NEXTASGN(elt,list,to,next) ((elt)->next)=(to) | ||||
| /* #define _PREV(elt,list,prev) ((elt)->prev) */ | ||||
| #define _PREVASGN(elt,list,to,prev) ((elt)->prev)=(to) | ||||
| #define _RS(list) | ||||
| #define _CASTASGN(a,b) (a)=(b) | ||||
| #endif | ||||
|  | ||||
| /****************************************************************************** | ||||
|  * The sort macro is an adaptation of Simon Tatham's O(n log(n)) mergesort    * | ||||
|  * Unwieldy variable names used here to avoid shadowing passed-in variables.  * | ||||
|  *****************************************************************************/ | ||||
| #define LL_SORT(list, cmp)                                                                     \ | ||||
|     LL_SORT2(list, cmp, next) | ||||
|  | ||||
| #define LL_SORT2(list, cmp, next)                                                              \ | ||||
| do {                                                                                           \ | ||||
|   LDECLTYPE(list) _ls_p;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_q;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_e;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_tail;                                                                    \ | ||||
|   int _ls_insize, _ls_nmerges, _ls_psize, _ls_qsize, _ls_i, _ls_looping;                       \ | ||||
|   if (list) {                                                                                  \ | ||||
|     _ls_insize = 1;                                                                            \ | ||||
|     _ls_looping = 1;                                                                           \ | ||||
|     while (_ls_looping) {                                                                      \ | ||||
|       _CASTASGN(_ls_p,list);                                                                   \ | ||||
|       list = NULL;                                                                             \ | ||||
|       _ls_tail = NULL;                                                                         \ | ||||
|       _ls_nmerges = 0;                                                                         \ | ||||
|       while (_ls_p) {                                                                          \ | ||||
|         _ls_nmerges++;                                                                         \ | ||||
|         _ls_q = _ls_p;                                                                         \ | ||||
|         _ls_psize = 0;                                                                         \ | ||||
|         for (_ls_i = 0; _ls_i < _ls_insize; _ls_i++) {                                         \ | ||||
|           _ls_psize++;                                                                         \ | ||||
|           _SV(_ls_q,list); _ls_q = _NEXT(_ls_q,list,next); _RS(list);                          \ | ||||
|           if (!_ls_q) break;                                                                   \ | ||||
|         }                                                                                      \ | ||||
|         _ls_qsize = _ls_insize;                                                                \ | ||||
|         while (_ls_psize > 0 || (_ls_qsize > 0 && _ls_q)) {                                    \ | ||||
|           if (_ls_psize == 0) {                                                                \ | ||||
|             _ls_e = _ls_q; _SV(_ls_q,list); _ls_q =                                            \ | ||||
|               _NEXT(_ls_q,list,next); _RS(list); _ls_qsize--;                                  \ | ||||
|           } else if (_ls_qsize == 0 || !_ls_q) {                                               \ | ||||
|             _ls_e = _ls_p; _SV(_ls_p,list); _ls_p =                                            \ | ||||
|               _NEXT(_ls_p,list,next); _RS(list); _ls_psize--;                                  \ | ||||
|           } else if (cmp(_ls_p,_ls_q) <= 0) {                                                  \ | ||||
|             _ls_e = _ls_p; _SV(_ls_p,list); _ls_p =                                            \ | ||||
|               _NEXT(_ls_p,list,next); _RS(list); _ls_psize--;                                  \ | ||||
|           } else {                                                                             \ | ||||
|             _ls_e = _ls_q; _SV(_ls_q,list); _ls_q =                                            \ | ||||
|               _NEXT(_ls_q,list,next); _RS(list); _ls_qsize--;                                  \ | ||||
|           }                                                                                    \ | ||||
|           if (_ls_tail) {                                                                      \ | ||||
|             _SV(_ls_tail,list); _NEXTASGN(_ls_tail,list,_ls_e,next); _RS(list);                \ | ||||
|           } else {                                                                             \ | ||||
|             _CASTASGN(list,_ls_e);                                                             \ | ||||
|           }                                                                                    \ | ||||
|           _ls_tail = _ls_e;                                                                    \ | ||||
|         }                                                                                      \ | ||||
|         _ls_p = _ls_q;                                                                         \ | ||||
|       }                                                                                        \ | ||||
|       if (_ls_tail) {                                                                          \ | ||||
|         _SV(_ls_tail,list); _NEXTASGN(_ls_tail,list,NULL,next); _RS(list);                     \ | ||||
|       }                                                                                        \ | ||||
|       if (_ls_nmerges <= 1) {                                                                  \ | ||||
|         _ls_looping=0;                                                                         \ | ||||
|       }                                                                                        \ | ||||
|       _ls_insize *= 2;                                                                         \ | ||||
|     }                                                                                          \ | ||||
|   }                                                                                            \ | ||||
| } while (0) | ||||
|  | ||||
|  | ||||
| #define DL_SORT(list, cmp)                                                                     \ | ||||
|     DL_SORT2(list, cmp, prev, next) | ||||
|  | ||||
| #define DL_SORT2(list, cmp, prev, next)                                                        \ | ||||
| do {                                                                                           \ | ||||
|   LDECLTYPE(list) _ls_p;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_q;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_e;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_tail;                                                                    \ | ||||
|   int _ls_insize, _ls_nmerges, _ls_psize, _ls_qsize, _ls_i, _ls_looping;                       \ | ||||
|   if (list) {                                                                                  \ | ||||
|     _ls_insize = 1;                                                                            \ | ||||
|     _ls_looping = 1;                                                                           \ | ||||
|     while (_ls_looping) {                                                                      \ | ||||
|       _CASTASGN(_ls_p,list);                                                                   \ | ||||
|       list = NULL;                                                                             \ | ||||
|       _ls_tail = NULL;                                                                         \ | ||||
|       _ls_nmerges = 0;                                                                         \ | ||||
|       while (_ls_p) {                                                                          \ | ||||
|         _ls_nmerges++;                                                                         \ | ||||
|         _ls_q = _ls_p;                                                                         \ | ||||
|         _ls_psize = 0;                                                                         \ | ||||
|         for (_ls_i = 0; _ls_i < _ls_insize; _ls_i++) {                                         \ | ||||
|           _ls_psize++;                                                                         \ | ||||
|           _SV(_ls_q,list); _ls_q = _NEXT(_ls_q,list,next); _RS(list);                          \ | ||||
|           if (!_ls_q) break;                                                                   \ | ||||
|         }                                                                                      \ | ||||
|         _ls_qsize = _ls_insize;                                                                \ | ||||
|         while (_ls_psize > 0 || (_ls_qsize > 0 && _ls_q)) {                                    \ | ||||
|           if (_ls_psize == 0) {                                                                \ | ||||
|             _ls_e = _ls_q; _SV(_ls_q,list); _ls_q =                                            \ | ||||
|               _NEXT(_ls_q,list,next); _RS(list); _ls_qsize--;                                  \ | ||||
|           } else if (_ls_qsize == 0 || !_ls_q) {                                               \ | ||||
|             _ls_e = _ls_p; _SV(_ls_p,list); _ls_p =                                            \ | ||||
|               _NEXT(_ls_p,list,next); _RS(list); _ls_psize--;                                  \ | ||||
|           } else if (cmp(_ls_p,_ls_q) <= 0) {                                                  \ | ||||
|             _ls_e = _ls_p; _SV(_ls_p,list); _ls_p =                                            \ | ||||
|               _NEXT(_ls_p,list,next); _RS(list); _ls_psize--;                                  \ | ||||
|           } else {                                                                             \ | ||||
|             _ls_e = _ls_q; _SV(_ls_q,list); _ls_q =                                            \ | ||||
|               _NEXT(_ls_q,list,next); _RS(list); _ls_qsize--;                                  \ | ||||
|           }                                                                                    \ | ||||
|           if (_ls_tail) {                                                                      \ | ||||
|             _SV(_ls_tail,list); _NEXTASGN(_ls_tail,list,_ls_e,next); _RS(list);                \ | ||||
|           } else {                                                                             \ | ||||
|             _CASTASGN(list,_ls_e);                                                             \ | ||||
|           }                                                                                    \ | ||||
|           _SV(_ls_e,list); _PREVASGN(_ls_e,list,_ls_tail,prev); _RS(list);                     \ | ||||
|           _ls_tail = _ls_e;                                                                    \ | ||||
|         }                                                                                      \ | ||||
|         _ls_p = _ls_q;                                                                         \ | ||||
|       }                                                                                        \ | ||||
|       _CASTASGN(list->prev, _ls_tail);                                                         \ | ||||
|       _SV(_ls_tail,list); _NEXTASGN(_ls_tail,list,NULL,next); _RS(list);                       \ | ||||
|       if (_ls_nmerges <= 1) {                                                                  \ | ||||
|         _ls_looping=0;                                                                         \ | ||||
|       }                                                                                        \ | ||||
|       _ls_insize *= 2;                                                                         \ | ||||
|     }                                                                                          \ | ||||
|   }                                                                                            \ | ||||
| } while (0) | ||||
|  | ||||
| #define CDL_SORT(list, cmp)                                                                    \ | ||||
|     CDL_SORT2(list, cmp, prev, next) | ||||
|  | ||||
| #define CDL_SORT2(list, cmp, prev, next)                                                       \ | ||||
| do {                                                                                           \ | ||||
|   LDECLTYPE(list) _ls_p;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_q;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_e;                                                                       \ | ||||
|   LDECLTYPE(list) _ls_tail;                                                                    \ | ||||
|   LDECLTYPE(list) _ls_oldhead;                                                                 \ | ||||
|   LDECLTYPE(list) _tmp;                                                                        \ | ||||
|   int _ls_insize, _ls_nmerges, _ls_psize, _ls_qsize, _ls_i, _ls_looping;                       \ | ||||
|   if (list) {                                                                                  \ | ||||
|     _ls_insize = 1;                                                                            \ | ||||
|     _ls_looping = 1;                                                                           \ | ||||
|     while (_ls_looping) {                                                                      \ | ||||
|       _CASTASGN(_ls_p,list);                                                                   \ | ||||
|       _CASTASGN(_ls_oldhead,list);                                                             \ | ||||
|       list = NULL;                                                                             \ | ||||
|       _ls_tail = NULL;                                                                         \ | ||||
|       _ls_nmerges = 0;                                                                         \ | ||||
|       while (_ls_p) {                                                                          \ | ||||
|         _ls_nmerges++;                                                                         \ | ||||
|         _ls_q = _ls_p;                                                                         \ | ||||
|         _ls_psize = 0;                                                                         \ | ||||
|         for (_ls_i = 0; _ls_i < _ls_insize; _ls_i++) {                                         \ | ||||
|           _ls_psize++;                                                                         \ | ||||
|           _SV(_ls_q,list);                                                                     \ | ||||
|           if (_NEXT(_ls_q,list,next) == _ls_oldhead) {                                         \ | ||||
|             _ls_q = NULL;                                                                      \ | ||||
|           } else {                                                                             \ | ||||
|             _ls_q = _NEXT(_ls_q,list,next);                                                    \ | ||||
|           }                                                                                    \ | ||||
|           _RS(list);                                                                           \ | ||||
|           if (!_ls_q) break;                                                                   \ | ||||
|         }                                                                                      \ | ||||
|         _ls_qsize = _ls_insize;                                                                \ | ||||
|         while (_ls_psize > 0 || (_ls_qsize > 0 && _ls_q)) {                                    \ | ||||
|           if (_ls_psize == 0) {                                                                \ | ||||
|             _ls_e = _ls_q; _SV(_ls_q,list); _ls_q =                                            \ | ||||
|               _NEXT(_ls_q,list,next); _RS(list); _ls_qsize--;                                  \ | ||||
|             if (_ls_q == _ls_oldhead) { _ls_q = NULL; }                                        \ | ||||
|           } else if (_ls_qsize == 0 || !_ls_q) {                                               \ | ||||
|             _ls_e = _ls_p; _SV(_ls_p,list); _ls_p =                                            \ | ||||
|               _NEXT(_ls_p,list,next); _RS(list); _ls_psize--;                                  \ | ||||
|             if (_ls_p == _ls_oldhead) { _ls_p = NULL; }                                        \ | ||||
|           } else if (cmp(_ls_p,_ls_q) <= 0) {                                                  \ | ||||
|             _ls_e = _ls_p; _SV(_ls_p,list); _ls_p =                                            \ | ||||
|               _NEXT(_ls_p,list,next); _RS(list); _ls_psize--;                                  \ | ||||
|             if (_ls_p == _ls_oldhead) { _ls_p = NULL; }                                        \ | ||||
|           } else {                                                                             \ | ||||
|             _ls_e = _ls_q; _SV(_ls_q,list); _ls_q =                                            \ | ||||
|               _NEXT(_ls_q,list,next); _RS(list); _ls_qsize--;                                  \ | ||||
|             if (_ls_q == _ls_oldhead) { _ls_q = NULL; }                                        \ | ||||
|           }                                                                                    \ | ||||
|           if (_ls_tail) {                                                                      \ | ||||
|             _SV(_ls_tail,list); _NEXTASGN(_ls_tail,list,_ls_e,next); _RS(list);                \ | ||||
|           } else {                                                                             \ | ||||
|             _CASTASGN(list,_ls_e);                                                             \ | ||||
|           }                                                                                    \ | ||||
|           _SV(_ls_e,list); _PREVASGN(_ls_e,list,_ls_tail,prev); _RS(list);                     \ | ||||
|           _ls_tail = _ls_e;                                                                    \ | ||||
|         }                                                                                      \ | ||||
|         _ls_p = _ls_q;                                                                         \ | ||||
|       }                                                                                        \ | ||||
|       _CASTASGN(list->prev,_ls_tail);                                                          \ | ||||
|       _CASTASGN(_tmp,list);                                                                    \ | ||||
|       _SV(_ls_tail,list); _NEXTASGN(_ls_tail,list,_tmp,next); _RS(list);                       \ | ||||
|       if (_ls_nmerges <= 1) {                                                                  \ | ||||
|         _ls_looping=0;                                                                         \ | ||||
|       }                                                                                        \ | ||||
|       _ls_insize *= 2;                                                                         \ | ||||
|     }                                                                                          \ | ||||
|   }                                                                                            \ | ||||
| } while (0) | ||||
|  | ||||
| /****************************************************************************** | ||||
|  * singly linked list macros (non-circular)                                   * | ||||
|  *****************************************************************************/ | ||||
| #define LL_PREPEND(head,add)                                                                   \ | ||||
|     LL_PREPEND2(head,add,next) | ||||
|  | ||||
| #define LL_PREPEND2(head,add,next)                                                             \ | ||||
| do {                                                                                           \ | ||||
|   (add)->next = head;                                                                          \ | ||||
|   head = add;                                                                                  \ | ||||
| } while (0) | ||||
|  | ||||
| #define LL_CONCAT(head1,head2)                                                                 \ | ||||
|     LL_CONCAT2(head1,head2,next) | ||||
|  | ||||
| #define LL_CONCAT2(head1,head2,next)                                                           \ | ||||
| do {                                                                                           \ | ||||
|   LDECLTYPE(head1) _tmp;                                                                       \ | ||||
|   if (head1) {                                                                                 \ | ||||
|     _tmp = head1;                                                                              \ | ||||
|     while (_tmp->next) { _tmp = _tmp->next; }                                                  \ | ||||
|     _tmp->next=(head2);                                                                        \ | ||||
|   } else {                                                                                     \ | ||||
|     (head1)=(head2);                                                                           \ | ||||
|   }                                                                                            \ | ||||
| } while (0) | ||||
|  | ||||
| #define LL_APPEND(head,add)                                                                    \ | ||||
|     LL_APPEND2(head,add,next) | ||||
|  | ||||
| #define LL_APPEND2(head,add,next)                                                              \ | ||||
| do {                                                                                           \ | ||||
|   LDECLTYPE(head) _tmp;                                                                        \ | ||||
|   (add)->next=NULL;                                                                            \ | ||||
|   if (head) {                                                                                  \ | ||||
|     _tmp = head;                                                                               \ | ||||
|     while (_tmp->next) { _tmp = _tmp->next; }                                                  \ | ||||
|     _tmp->next=(add);                                                                          \ | ||||
|   } else {                                                                                     \ | ||||
|     (head)=(add);                                                                              \ | ||||
|   }                                                                                            \ | ||||
| } while (0) | ||||
|  | ||||
| #define LL_DELETE(head,del)                                                                    \ | ||||
|     LL_DELETE2(head,del,next) | ||||
|  | ||||
| #define LL_DELETE2(head,del,next)                                                              \ | ||||
| do {                                                                                           \ | ||||
|   LDECLTYPE(head) _tmp;                                                                        \ | ||||
|   if ((head) == (del)) {                                                                       \ | ||||
|     (head)=(head)->next;                                                                       \ | ||||
|   } else {                                                                                     \ | ||||
|     _tmp = head;                                                                               \ | ||||
|     while (_tmp->next && (_tmp->next != (del))) {                                              \ | ||||
|       _tmp = _tmp->next;                                                                       \ | ||||
|     }                                                                                          \ | ||||
|     if (_tmp->next) {                                                                          \ | ||||
|       _tmp->next = ((del)->next);                                                              \ | ||||
|     }                                                                                          \ | ||||
|   }                                                                                            \ | ||||
| } while (0) | ||||
|  | ||||
| /* Here are VS2008 replacements for LL_APPEND and LL_DELETE */ | ||||
| #define LL_APPEND_VS2008(head,add)                                                             \ | ||||
|     LL_APPEND2_VS2008(head,add,next) | ||||
|  | ||||
| #define LL_APPEND2_VS2008(head,add,next)                                                       \ | ||||
| do {                                                                                           \ | ||||
|   if (head) {                                                                                  \ | ||||
|     (add)->next = head;     /* use add->next as a temp variable */                             \ | ||||
|     while ((add)->next->next) { (add)->next = (add)->next->next; }                             \ | ||||
|     (add)->next->next=(add);                                                                   \ | ||||
|   } else {                                                                                     \ | ||||
|     (head)=(add);                                                                              \ | ||||
|   }                                                                                            \ | ||||
|   (add)->next=NULL;                                                                            \ | ||||
| } while (0) | ||||
|  | ||||
| #define LL_DELETE_VS2008(head,del)                                                             \ | ||||
|     LL_DELETE2_VS2008(head,del,next) | ||||
|  | ||||
| #define LL_DELETE2_VS2008(head,del,next)                                                       \ | ||||
| do {                                                                                           \ | ||||
|   if ((head) == (del)) {                                                                       \ | ||||
|     (head)=(head)->next;                                                                       \ | ||||
|   } else {                                                                                     \ | ||||
|     char *_tmp = (char*)(head);                                                                \ | ||||
|     while ((head)->next && ((head)->next != (del))) {                                          \ | ||||
|       head = (head)->next;                                                                     \ | ||||
|     }                                                                                          \ | ||||
|     if ((head)->next) {                                                                        \ | ||||
|       (head)->next = ((del)->next);                                                            \ | ||||
|     }                                                                                          \ | ||||
|     {                                                                                          \ | ||||
|       char **_head_alias = (char**)&(head);                                                    \ | ||||
|       *_head_alias = _tmp;                                                                     \ | ||||
|     }                                                                                          \ | ||||
|   }                                                                                            \ | ||||
| } while (0) | ||||
| #ifdef NO_DECLTYPE | ||||
| #undef LL_APPEND | ||||
| #define LL_APPEND LL_APPEND_VS2008 | ||||
| #undef LL_DELETE | ||||
| #define LL_DELETE LL_DELETE_VS2008 | ||||
| #undef LL_DELETE2 | ||||
| #define LL_DELETE2 LL_DELETE2_VS2008 | ||||
| #undef LL_APPEND2 | ||||
| #define LL_APPEND2 LL_APPEND2_VS2008 | ||||
| #undef LL_CONCAT /* no LL_CONCAT_VS2008 */ | ||||
| #undef DL_CONCAT /* no DL_CONCAT_VS2008 */ | ||||
| #endif | ||||
| /* end VS2008 replacements */ | ||||
|  | ||||
| #define LL_COUNT(head,el,counter)                                                              \ | ||||
|     LL_COUNT2(head,el,counter,next)                                                            \ | ||||
|  | ||||
| #define LL_COUNT2(head,el,counter,next)                                                        \ | ||||
| {                                                                                              \ | ||||
|     counter = 0;                                                                               \ | ||||
|     LL_FOREACH2(head,el,next){ ++counter; }                                                    \ | ||||
| } | ||||
|  | ||||
| #define LL_FOREACH(head,el)                                                                    \ | ||||
|     LL_FOREACH2(head,el,next) | ||||
|  | ||||
| #define LL_FOREACH2(head,el,next)                                                              \ | ||||
|     for(el=head;el;el=(el)->next) | ||||
|  | ||||
| #define LL_FOREACH_SAFE(head,el,tmp)                                                           \ | ||||
|     LL_FOREACH_SAFE2(head,el,tmp,next) | ||||
|  | ||||
| #define LL_FOREACH_SAFE2(head,el,tmp,next)                                                     \ | ||||
|   for((el)=(head);(el) && (tmp = (el)->next, 1); (el) = tmp) | ||||
|  | ||||
| #define LL_SEARCH_SCALAR(head,out,field,val)                                                   \ | ||||
|     LL_SEARCH_SCALAR2(head,out,field,val,next) | ||||
|  | ||||
| #define LL_SEARCH_SCALAR2(head,out,field,val,next)                                             \ | ||||
| do {                                                                                           \ | ||||
|     LL_FOREACH2(head,out,next) {                                                               \ | ||||
|       if ((out)->field == (val)) break;                                                        \ | ||||
|     }                                                                                          \ | ||||
| } while(0)  | ||||
|  | ||||
| #define LL_SEARCH(head,out,elt,cmp)                                                            \ | ||||
|     LL_SEARCH2(head,out,elt,cmp,next) | ||||
|  | ||||
| #define LL_SEARCH2(head,out,elt,cmp,next)                                                      \ | ||||
| do {                                                                                           \ | ||||
|     LL_FOREACH2(head,out,next) {                                                               \ | ||||
|       if ((cmp(out,elt))==0) break;                                                            \ | ||||
|     }                                                                                          \ | ||||
| } while(0)  | ||||
|  | ||||
| #define LL_REPLACE_ELEM(head, el, add)                                                         \ | ||||
| do {                                                                                           \ | ||||
|  LDECLTYPE(head) _tmp;                                                                         \ | ||||
|  assert(head != NULL);                                                                         \ | ||||
|  assert(el != NULL);                                                                           \ | ||||
|  assert(add != NULL);                                                                          \ | ||||
|  (add)->next = (el)->next;                                                                     \ | ||||
|  if ((head) == (el)) {                                                                         \ | ||||
|   (head) = (add);                                                                              \ | ||||
|  } else {                                                                                      \ | ||||
|   _tmp = head;                                                                                 \ | ||||
|   while (_tmp->next && (_tmp->next != (el))) {                                                 \ | ||||
|    _tmp = _tmp->next;                                                                          \ | ||||
|   }                                                                                            \ | ||||
|   if (_tmp->next) {                                                                            \ | ||||
|     _tmp->next = (add);                                                                        \ | ||||
|   }                                                                                            \ | ||||
|  }                                                                                             \ | ||||
| } while (0) | ||||
|  | ||||
| #define LL_PREPEND_ELEM(head, el, add)                                                         \ | ||||
| do {                                                                                           \ | ||||
|  LDECLTYPE(head) _tmp;                                                                         \ | ||||
|  assert(head != NULL);                                                                         \ | ||||
|  assert(el != NULL);                                                                           \ | ||||
|  assert(add != NULL);                                                                          \ | ||||
|  (add)->next = (el);                                                                           \ | ||||
|  if ((head) == (el)) {                                                                         \ | ||||
|   (head) = (add);                                                                              \ | ||||
|  } else {                                                                                      \ | ||||
|   _tmp = head;                                                                                 \ | ||||
|   while (_tmp->next && (_tmp->next != (el))) {                                                 \ | ||||
|    _tmp = _tmp->next;                                                                          \ | ||||
|   }                                                                                            \ | ||||
|   if (_tmp->next) {                                                                            \ | ||||
|     _tmp->next = (add);                                                                        \ | ||||
|   }                                                                                            \ | ||||
|  }                                                                                             \ | ||||
| } while (0)                                                                                    \ | ||||
|  | ||||
|  | ||||
| /****************************************************************************** | ||||
|  * doubly linked list macros (non-circular)                                   * | ||||
|  *****************************************************************************/ | ||||
| #define DL_PREPEND(head,add)                                                                   \ | ||||
|     DL_PREPEND2(head,add,prev,next) | ||||
|  | ||||
| #define DL_PREPEND2(head,add,prev,next)                                                        \ | ||||
| do {                                                                                           \ | ||||
|  (add)->next = head;                                                                           \ | ||||
|  if (head) {                                                                                   \ | ||||
|    (add)->prev = (head)->prev;                                                                 \ | ||||
|    (head)->prev = (add);                                                                       \ | ||||
|  } else {                                                                                      \ | ||||
|    (add)->prev = (add);                                                                        \ | ||||
|  }                                                                                             \ | ||||
|  (head) = (add);                                                                               \ | ||||
| } while (0) | ||||
|  | ||||
| #define DL_APPEND(head,add)                                                                    \ | ||||
|     DL_APPEND2(head,add,prev,next) | ||||
|  | ||||
| #define DL_APPEND2(head,add,prev,next)                                                         \ | ||||
| do {                                                                                           \ | ||||
|   if (head) {                                                                                  \ | ||||
|       (add)->prev = (head)->prev;                                                              \ | ||||
|       (head)->prev->next = (add);                                                              \ | ||||
|       (head)->prev = (add);                                                                    \ | ||||
|       (add)->next = NULL;                                                                      \ | ||||
|   } else {                                                                                     \ | ||||
|       (head)=(add);                                                                            \ | ||||
|       (head)->prev = (head);                                                                   \ | ||||
|       (head)->next = NULL;                                                                     \ | ||||
|   }                                                                                            \ | ||||
| } while (0)  | ||||
|  | ||||
| #define DL_CONCAT(head1,head2)                                                                 \ | ||||
|     DL_CONCAT2(head1,head2,prev,next) | ||||
|  | ||||
| #define DL_CONCAT2(head1,head2,prev,next)                                                      \ | ||||
| do {                                                                                           \ | ||||
|   LDECLTYPE(head1) _tmp;                                                                       \ | ||||
|   if (head2) {                                                                                 \ | ||||
|     if (head1) {                                                                               \ | ||||
|         _tmp = (head2)->prev;                                                                  \ | ||||
|         (head2)->prev = (head1)->prev;                                                         \ | ||||
|         (head1)->prev->next = (head2);                                                         \ | ||||
|         (head1)->prev = _tmp;                                                                  \ | ||||
|     } else {                                                                                   \ | ||||
|         (head1)=(head2);                                                                       \ | ||||
|     }                                                                                          \ | ||||
|   }                                                                                            \ | ||||
| } while (0)  | ||||
|  | ||||
| #define DL_DELETE(head,del)                                                                    \ | ||||
|     DL_DELETE2(head,del,prev,next) | ||||
|  | ||||
| #define DL_DELETE2(head,del,prev,next)                                                         \ | ||||
| do {                                                                                           \ | ||||
|   assert((del)->prev != NULL);                                                                 \ | ||||
|   if ((del)->prev == (del)) {                                                                  \ | ||||
|       (head)=NULL;                                                                             \ | ||||
|   } else if ((del)==(head)) {                                                                  \ | ||||
|       (del)->next->prev = (del)->prev;                                                         \ | ||||
|       (head) = (del)->next;                                                                    \ | ||||
|   } else {                                                                                     \ | ||||
|       (del)->prev->next = (del)->next;                                                         \ | ||||
|       if ((del)->next) {                                                                       \ | ||||
|           (del)->next->prev = (del)->prev;                                                     \ | ||||
|       } else {                                                                                 \ | ||||
|           (head)->prev = (del)->prev;                                                          \ | ||||
|       }                                                                                        \ | ||||
|   }                                                                                            \ | ||||
| } while (0)  | ||||
|  | ||||
| #define DL_COUNT(head,el,counter)                                                              \ | ||||
|     DL_COUNT2(head,el,counter,next)                                                            \ | ||||
|  | ||||
| #define DL_COUNT2(head,el,counter,next)                                                        \ | ||||
| {                                                                                              \ | ||||
|     counter = 0;                                                                               \ | ||||
|     DL_FOREACH2(head,el,next){ ++counter; }                                                    \ | ||||
| } | ||||
|  | ||||
| #define DL_FOREACH(head,el)                                                                    \ | ||||
|     DL_FOREACH2(head,el,next) | ||||
|  | ||||
| #define DL_FOREACH2(head,el,next)                                                              \ | ||||
|     for(el=head;el;el=(el)->next) | ||||
|  | ||||
| /* this version is safe for deleting the elements during iteration */ | ||||
| #define DL_FOREACH_SAFE(head,el,tmp)                                                           \ | ||||
|     DL_FOREACH_SAFE2(head,el,tmp,next) | ||||
|  | ||||
| #define DL_FOREACH_SAFE2(head,el,tmp,next)                                                     \ | ||||
|   for((el)=(head);(el) && (tmp = (el)->next, 1); (el) = tmp) | ||||
|  | ||||
| /* these are identical to their singly-linked list counterparts */ | ||||
| #define DL_SEARCH_SCALAR LL_SEARCH_SCALAR | ||||
| #define DL_SEARCH LL_SEARCH | ||||
| #define DL_SEARCH_SCALAR2 LL_SEARCH_SCALAR2 | ||||
| #define DL_SEARCH2 LL_SEARCH2 | ||||
|  | ||||
| #define DL_REPLACE_ELEM(head, el, add)                                                         \ | ||||
| do {                                                                                           \ | ||||
|  assert(head != NULL);                                                                         \ | ||||
|  assert(el != NULL);                                                                           \ | ||||
|  assert(add != NULL);                                                                          \ | ||||
|  if ((head) == (el)) {                                                                         \ | ||||
|   (head) = (add);                                                                              \ | ||||
|   (add)->next = (el)->next;                                                                    \ | ||||
|   if ((el)->next == NULL) {                                                                    \ | ||||
|    (add)->prev = (add);                                                                        \ | ||||
|   } else {                                                                                     \ | ||||
|    (add)->prev = (el)->prev;                                                                   \ | ||||
|    (add)->next->prev = (add);                                                                  \ | ||||
|   }                                                                                            \ | ||||
|  } else {                                                                                      \ | ||||
|   (add)->next = (el)->next;                                                                    \ | ||||
|   (add)->prev = (el)->prev;                                                                    \ | ||||
|   (add)->prev->next = (add);                                                                   \ | ||||
|   if ((el)->next == NULL) {                                                                    \ | ||||
|    (head)->prev = (add);                                                                       \ | ||||
|   } else {                                                                                     \ | ||||
|    (add)->next->prev = (add);                                                                  \ | ||||
|   }                                                                                            \ | ||||
|  }                                                                                             \ | ||||
| } while (0) | ||||
|  | ||||
| #define DL_PREPEND_ELEM(head, el, add)                                                         \ | ||||
| do {                                                                                           \ | ||||
|  assert(head != NULL);                                                                         \ | ||||
|  assert(el != NULL);                                                                           \ | ||||
|  assert(add != NULL);                                                                          \ | ||||
|  (add)->next = (el);                                                                           \ | ||||
|  (add)->prev = (el)->prev;                                                                     \ | ||||
|  (el)->prev = (add);                                                                           \ | ||||
|  if ((head) == (el)) {                                                                         \ | ||||
|   (head) = (add);                                                                              \ | ||||
|  } else {                                                                                      \ | ||||
|   (add)->prev->next = (add);                                                                   \ | ||||
|  }                                                                                             \ | ||||
| } while (0)                                                                                    \ | ||||
|  | ||||
|  | ||||
| /****************************************************************************** | ||||
|  * circular doubly linked list macros                                         * | ||||
|  *****************************************************************************/ | ||||
| #define CDL_PREPEND(head,add)                                                                  \ | ||||
|     CDL_PREPEND2(head,add,prev,next) | ||||
|  | ||||
| #define CDL_PREPEND2(head,add,prev,next)                                                       \ | ||||
| do {                                                                                           \ | ||||
|  if (head) {                                                                                   \ | ||||
|    (add)->prev = (head)->prev;                                                                 \ | ||||
|    (add)->next = (head);                                                                       \ | ||||
|    (head)->prev = (add);                                                                       \ | ||||
|    (add)->prev->next = (add);                                                                  \ | ||||
|  } else {                                                                                      \ | ||||
|    (add)->prev = (add);                                                                        \ | ||||
|    (add)->next = (add);                                                                        \ | ||||
|  }                                                                                             \ | ||||
| (head)=(add);                                                                                  \ | ||||
| } while (0) | ||||
|  | ||||
| #define CDL_DELETE(head,del)                                                                   \ | ||||
|     CDL_DELETE2(head,del,prev,next) | ||||
|  | ||||
| #define CDL_DELETE2(head,del,prev,next)                                                        \ | ||||
| do {                                                                                           \ | ||||
|   if ( ((head)==(del)) && ((head)->next == (head))) {                                          \ | ||||
|       (head) = 0L;                                                                             \ | ||||
|   } else {                                                                                     \ | ||||
|      (del)->next->prev = (del)->prev;                                                          \ | ||||
|      (del)->prev->next = (del)->next;                                                          \ | ||||
|      if ((del) == (head)) (head)=(del)->next;                                                  \ | ||||
|   }                                                                                            \ | ||||
| } while (0)  | ||||
|  | ||||
| #define CDL_COUNT(head,el,counter)                                                             \ | ||||
|     CDL_COUNT2(head,el,counter,next)                                                           \ | ||||
|  | ||||
| #define CDL_COUNT2(head, el, counter,next)                                                     \ | ||||
| {                                                                                              \ | ||||
|     counter = 0;                                                                               \ | ||||
|     CDL_FOREACH2(head,el,next){ ++counter; }                                                   \ | ||||
| } | ||||
|  | ||||
| #define CDL_FOREACH(head,el)                                                                   \ | ||||
|     CDL_FOREACH2(head,el,next) | ||||
|  | ||||
| #define CDL_FOREACH2(head,el,next)                                                             \ | ||||
|     for(el=head;el;el=((el)->next==head ? 0L : (el)->next))  | ||||
|  | ||||
| #define CDL_FOREACH_SAFE(head,el,tmp1,tmp2)                                                    \ | ||||
|     CDL_FOREACH_SAFE2(head,el,tmp1,tmp2,prev,next) | ||||
|  | ||||
| #define CDL_FOREACH_SAFE2(head,el,tmp1,tmp2,prev,next)                                         \ | ||||
|   for((el)=(head), ((tmp1)=(head)?((head)->prev):NULL);                                        \ | ||||
|       (el) && ((tmp2)=(el)->next, 1);                                                          \ | ||||
|       ((el) = (((el)==(tmp1)) ? 0L : (tmp2)))) | ||||
|  | ||||
| #define CDL_SEARCH_SCALAR(head,out,field,val)                                                  \ | ||||
|     CDL_SEARCH_SCALAR2(head,out,field,val,next) | ||||
|  | ||||
| #define CDL_SEARCH_SCALAR2(head,out,field,val,next)                                            \ | ||||
| do {                                                                                           \ | ||||
|     CDL_FOREACH2(head,out,next) {                                                              \ | ||||
|       if ((out)->field == (val)) break;                                                        \ | ||||
|     }                                                                                          \ | ||||
| } while(0)  | ||||
|  | ||||
| #define CDL_SEARCH(head,out,elt,cmp)                                                           \ | ||||
|     CDL_SEARCH2(head,out,elt,cmp,next) | ||||
|  | ||||
| #define CDL_SEARCH2(head,out,elt,cmp,next)                                                     \ | ||||
| do {                                                                                           \ | ||||
|     CDL_FOREACH2(head,out,next) {                                                              \ | ||||
|       if ((cmp(out,elt))==0) break;                                                            \ | ||||
|     }                                                                                          \ | ||||
| } while(0)  | ||||
|  | ||||
| #define CDL_REPLACE_ELEM(head, el, add)                                                        \ | ||||
| do {                                                                                           \ | ||||
|  assert(head != NULL);                                                                         \ | ||||
|  assert(el != NULL);                                                                           \ | ||||
|  assert(add != NULL);                                                                          \ | ||||
|  if ((el)->next == (el)) {                                                                     \ | ||||
|   (add)->next = (add);                                                                         \ | ||||
|   (add)->prev = (add);                                                                         \ | ||||
|   (head) = (add);                                                                              \ | ||||
|  } else {                                                                                      \ | ||||
|   (add)->next = (el)->next;                                                                    \ | ||||
|   (add)->prev = (el)->prev;                                                                    \ | ||||
|   (add)->next->prev = (add);                                                                   \ | ||||
|   (add)->prev->next = (add);                                                                   \ | ||||
|   if ((head) == (el)) {                                                                        \ | ||||
|    (head) = (add);                                                                             \ | ||||
|   }                                                                                            \ | ||||
|  }                                                                                             \ | ||||
| } while (0) | ||||
|  | ||||
| #define CDL_PREPEND_ELEM(head, el, add)                                                        \ | ||||
| do {                                                                                           \ | ||||
|  assert(head != NULL);                                                                         \ | ||||
|  assert(el != NULL);                                                                           \ | ||||
|  assert(add != NULL);                                                                          \ | ||||
|  (add)->next = (el);                                                                           \ | ||||
|  (add)->prev = (el)->prev;                                                                     \ | ||||
|  (el)->prev = (add);                                                                           \ | ||||
|  (add)->prev->next = (add);                                                                    \ | ||||
|  if ((head) == (el)) {                                                                         \ | ||||
|   (head) = (add);                                                                              \ | ||||
|  }                                                                                             \ | ||||
| } while (0)                                                                                    \ | ||||
|  | ||||
| #endif /* UTLIST_H */ | ||||
|  | ||||
							
								
								
									
										392
									
								
								src/ut/utstring.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										392
									
								
								src/ut/utstring.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,392 @@ | ||||
| /* | ||||
| Copyright (c) 2008-2014, Troy D. Hanson   http://troydhanson.github.com/uthash/ | ||||
| All rights reserved. | ||||
|  | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions are met: | ||||
|  | ||||
|     * Redistributions of source code must retain the above copyright | ||||
|       notice, this list of conditions and the following disclaimer. | ||||
|  | ||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS | ||||
| IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED | ||||
| TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A | ||||
| PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER | ||||
| OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | ||||
| EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | ||||
| PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | ||||
| PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF | ||||
| LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | ||||
| NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||||
| SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
| */ | ||||
|  | ||||
| /* a dynamic string implementation using macros  | ||||
|  */ | ||||
| #ifndef UTSTRING_H | ||||
| #define UTSTRING_H | ||||
|  | ||||
| #define UTSTRING_VERSION 1.9.9 | ||||
|  | ||||
| #ifdef __GNUC__ | ||||
| #define _UNUSED_ __attribute__ ((__unused__))  | ||||
| #else | ||||
| #define _UNUSED_  | ||||
| #endif | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <stdarg.h> | ||||
| #define oom() exit(-1) | ||||
|  | ||||
| typedef struct { | ||||
|     char *d; | ||||
|     size_t n; /* allocd size */ | ||||
|     size_t i; /* index of first unused byte */ | ||||
| } UT_string; | ||||
|  | ||||
| #define utstring_reserve(s,amt)                            \ | ||||
| do {                                                       \ | ||||
|   if (((s)->n - (s)->i) < (size_t)(amt)) {                 \ | ||||
|      (s)->d = (char*)realloc((s)->d, (s)->n + amt);        \ | ||||
|      if ((s)->d == NULL) oom();                            \ | ||||
|      (s)->n += amt;                                        \ | ||||
|   }                                                        \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_init(s)                                   \ | ||||
| do {                                                       \ | ||||
|   (s)->n = 0; (s)->i = 0; (s)->d = NULL;                   \ | ||||
|   utstring_reserve(s,100);                                 \ | ||||
|   (s)->d[0] = '\0'; \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_done(s)                                   \ | ||||
| do {                                                       \ | ||||
|   if ((s)->d != NULL) free((s)->d);                        \ | ||||
|   (s)->n = 0;                                              \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_free(s)                                   \ | ||||
| do {                                                       \ | ||||
|   utstring_done(s);                                        \ | ||||
|   free(s);                                                 \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_new(s)                                    \ | ||||
| do {                                                       \ | ||||
|    s = (UT_string*)calloc(sizeof(UT_string),1);            \ | ||||
|    if (!s) oom();                                          \ | ||||
|    utstring_init(s);                                       \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_renew(s)                                  \ | ||||
| do {                                                       \ | ||||
|    if (s) {                                                \ | ||||
|      utstring_clear(s);                                    \ | ||||
|    } else {                                                \ | ||||
|      utstring_new(s);                                      \ | ||||
|    }                                                       \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_clear(s)                                  \ | ||||
| do {                                                       \ | ||||
|   (s)->i = 0;                                              \ | ||||
|   (s)->d[0] = '\0';                                        \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_bincpy(s,b,l)                             \ | ||||
| do {                                                       \ | ||||
|   utstring_reserve((s),(l)+1);                               \ | ||||
|   if (l) memcpy(&(s)->d[(s)->i], b, l);                    \ | ||||
|   (s)->i += l;                                               \ | ||||
|   (s)->d[(s)->i]='\0';                                         \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_concat(dst,src)                                 \ | ||||
| do {                                                             \ | ||||
|   utstring_reserve((dst),((src)->i)+1);                          \ | ||||
|   if ((src)->i) memcpy(&(dst)->d[(dst)->i], (src)->d, (src)->i); \ | ||||
|   (dst)->i += (src)->i;                                          \ | ||||
|   (dst)->d[(dst)->i]='\0';                                       \ | ||||
| } while(0) | ||||
|  | ||||
| #define utstring_len(s) ((unsigned)((s)->i)) | ||||
|  | ||||
| #define utstring_body(s) ((s)->d) | ||||
|  | ||||
| _UNUSED_ static void utstring_printf_va(UT_string *s, const char *fmt, va_list ap) { | ||||
|    int n; | ||||
|    va_list cp; | ||||
|    while (1) { | ||||
| #ifdef _WIN32 | ||||
|       cp = ap; | ||||
| #else | ||||
|       va_copy(cp, ap); | ||||
| #endif | ||||
|       n = vsnprintf (&s->d[s->i], s->n-s->i, fmt, cp); | ||||
|       va_end(cp); | ||||
|  | ||||
|       if ((n > -1) && ((size_t) n < (s->n-s->i))) { | ||||
|         s->i += n; | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       /* Else try again with more space. */ | ||||
|       if (n > -1) utstring_reserve(s,n+1); /* exact */ | ||||
|       else utstring_reserve(s,(s->n)*2);   /* 2x */ | ||||
|    } | ||||
| } | ||||
| #ifdef __GNUC__ | ||||
| /* support printf format checking (2=the format string, 3=start of varargs) */ | ||||
| static void utstring_printf(UT_string *s, const char *fmt, ...) | ||||
|   __attribute__ (( format( printf, 2, 3) )); | ||||
| #endif | ||||
| _UNUSED_ static void utstring_printf(UT_string *s, const char *fmt, ...) { | ||||
|    va_list ap; | ||||
|    va_start(ap,fmt); | ||||
|    utstring_printf_va(s,fmt,ap); | ||||
|    va_end(ap); | ||||
| } | ||||
|  | ||||
| /******************************************************************************* | ||||
|  * begin substring search functions                                            * | ||||
|  ******************************************************************************/ | ||||
| /* Build KMP table from left to right. */ | ||||
| _UNUSED_ static void _utstring_BuildTable( | ||||
|     const char *P_Needle,  | ||||
|     size_t P_NeedleLen,  | ||||
|     long *P_KMP_Table) | ||||
| { | ||||
|     long i, j; | ||||
|  | ||||
|     i = 0; | ||||
|     j = i - 1; | ||||
|     P_KMP_Table[i] = j; | ||||
|     while (i < P_NeedleLen) | ||||
|     { | ||||
|         while ( (j > -1) && (P_Needle[i] != P_Needle[j]) ) | ||||
|         { | ||||
|            j = P_KMP_Table[j]; | ||||
|         } | ||||
|         i++; | ||||
|         j++; | ||||
|         if (i < P_NeedleLen) | ||||
|         { | ||||
|             if (P_Needle[i] == P_Needle[j]) | ||||
|             { | ||||
|                 P_KMP_Table[i] = P_KMP_Table[j]; | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 P_KMP_Table[i] = j; | ||||
|             } | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             P_KMP_Table[i] = j; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Build KMP table from right to left. */ | ||||
| _UNUSED_ static void _utstring_BuildTableR( | ||||
|     const char *P_Needle,  | ||||
|     size_t P_NeedleLen,  | ||||
|     long *P_KMP_Table) | ||||
| { | ||||
|     long i, j; | ||||
|  | ||||
|     i = P_NeedleLen - 1; | ||||
|     j = i + 1; | ||||
|     P_KMP_Table[i + 1] = j; | ||||
|     while (i >= 0) | ||||
|     { | ||||
|         while ( (j < P_NeedleLen) && (P_Needle[i] != P_Needle[j]) ) | ||||
|         { | ||||
|            j = P_KMP_Table[j + 1]; | ||||
|         } | ||||
|         i--; | ||||
|         j--; | ||||
|         if (i >= 0) | ||||
|         { | ||||
|             if (P_Needle[i] == P_Needle[j]) | ||||
|             { | ||||
|                 P_KMP_Table[i + 1] = P_KMP_Table[j + 1]; | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 P_KMP_Table[i + 1] = j; | ||||
|             } | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             P_KMP_Table[i + 1] = j; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Search data from left to right. ( Multiple search mode. ) */ | ||||
| _UNUSED_ static long _utstring_find( | ||||
|     const char *P_Haystack,  | ||||
|     size_t P_HaystackLen,  | ||||
|     const char *P_Needle,  | ||||
|     size_t P_NeedleLen,  | ||||
|     long *P_KMP_Table) | ||||
| { | ||||
|     long i, j; | ||||
|     long V_FindPosition = -1; | ||||
|  | ||||
|     /* Search from left to right. */ | ||||
|     i = j = 0; | ||||
|     while ( (j < (int)P_HaystackLen) && (((P_HaystackLen - j) + i) >= P_NeedleLen) ) | ||||
|     { | ||||
|         while ( (i > -1) && (P_Needle[i] != P_Haystack[j]) ) | ||||
|         { | ||||
|             i = P_KMP_Table[i]; | ||||
|         } | ||||
|         i++; | ||||
|         j++; | ||||
|         if (i >= (int)P_NeedleLen) | ||||
|         { | ||||
|             /* Found. */ | ||||
|             V_FindPosition = j - i; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return V_FindPosition; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Search data from right to left. ( Multiple search mode. ) */ | ||||
| _UNUSED_ static long _utstring_findR( | ||||
|     const char *P_Haystack,  | ||||
|     size_t P_HaystackLen,  | ||||
|     const char *P_Needle,  | ||||
|     size_t P_NeedleLen,  | ||||
|     long *P_KMP_Table) | ||||
| { | ||||
|     long i, j; | ||||
|     long V_FindPosition = -1; | ||||
|  | ||||
|     /* Search from right to left. */ | ||||
|     j = (P_HaystackLen - 1); | ||||
|     i = (P_NeedleLen - 1); | ||||
|     while ( (j >= 0) && (j >= i) ) | ||||
|     { | ||||
|         while ( (i < (int)P_NeedleLen) && (P_Needle[i] != P_Haystack[j]) ) | ||||
|         { | ||||
|             i = P_KMP_Table[i + 1]; | ||||
|         } | ||||
|         i--; | ||||
|         j--; | ||||
|         if (i < 0) | ||||
|         { | ||||
|             /* Found. */ | ||||
|             V_FindPosition = j + 1; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return V_FindPosition; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Search data from left to right. ( One time search mode. ) */ | ||||
| _UNUSED_ static long utstring_find( | ||||
|     UT_string *s,  | ||||
|     long P_StartPosition,   /* Start from 0. -1 means last position. */ | ||||
|     const char *P_Needle,  | ||||
|     size_t P_NeedleLen) | ||||
| { | ||||
|     long V_StartPosition; | ||||
|     long V_HaystackLen; | ||||
|     long *V_KMP_Table; | ||||
|     long V_FindPosition = -1; | ||||
|  | ||||
|     if (P_StartPosition < 0) | ||||
|     { | ||||
|         V_StartPosition = s->i + P_StartPosition; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         V_StartPosition = P_StartPosition; | ||||
|     } | ||||
|     V_HaystackLen = s->i - V_StartPosition; | ||||
|     if ( (V_HaystackLen >= P_NeedleLen) && (P_NeedleLen > 0) ) | ||||
|     { | ||||
|         V_KMP_Table = (long *)malloc(sizeof(long) * (P_NeedleLen + 1)); | ||||
|         if (V_KMP_Table != NULL) | ||||
|         { | ||||
|             _utstring_BuildTable(P_Needle, P_NeedleLen, V_KMP_Table); | ||||
|  | ||||
|             V_FindPosition = _utstring_find(s->d + V_StartPosition,  | ||||
|                                             V_HaystackLen,  | ||||
|                                             P_Needle,  | ||||
|                                             P_NeedleLen,  | ||||
|                                             V_KMP_Table); | ||||
|             if (V_FindPosition >= 0) | ||||
|             { | ||||
|                 V_FindPosition += V_StartPosition; | ||||
|             } | ||||
|  | ||||
|             free(V_KMP_Table); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return V_FindPosition; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Search data from right to left. ( One time search mode. ) */ | ||||
| _UNUSED_ static long utstring_findR( | ||||
|     UT_string *s,  | ||||
|     long P_StartPosition,   /* Start from 0. -1 means last position. */ | ||||
|     const char *P_Needle,  | ||||
|     size_t P_NeedleLen) | ||||
| { | ||||
|     long V_StartPosition; | ||||
|     long V_HaystackLen; | ||||
|     long *V_KMP_Table; | ||||
|     long V_FindPosition = -1; | ||||
|  | ||||
|     if (P_StartPosition < 0) | ||||
|     { | ||||
|         V_StartPosition = s->i + P_StartPosition; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         V_StartPosition = P_StartPosition; | ||||
|     } | ||||
|     V_HaystackLen = V_StartPosition + 1; | ||||
|     if ( (V_HaystackLen >= P_NeedleLen) && (P_NeedleLen > 0) ) | ||||
|     { | ||||
|         V_KMP_Table = (long *)malloc(sizeof(long) * (P_NeedleLen + 1)); | ||||
|         if (V_KMP_Table != NULL) | ||||
|         { | ||||
|             _utstring_BuildTableR(P_Needle, P_NeedleLen, V_KMP_Table); | ||||
|  | ||||
|             V_FindPosition = _utstring_findR(s->d,  | ||||
|                                              V_HaystackLen,  | ||||
|                                              P_Needle,  | ||||
|                                              P_NeedleLen,  | ||||
|                                              V_KMP_Table); | ||||
|  | ||||
|             free(V_KMP_Table); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return V_FindPosition; | ||||
| } | ||||
| /******************************************************************************* | ||||
|  * end substring search functions                                              * | ||||
|  ******************************************************************************/ | ||||
|  | ||||
| #endif /* UTSTRING_H */ | ||||
		Reference in New Issue
	
	Block a user