postgis/liblwgeom/stringbuffer.c
Paul Ramsey c23debb38a Convert GeoJSON output to use stringbuffer instead of count-n-fill procedure. Much less code.
Performance for large-amounts-of-small objects is about 10% faster. Performance for larger
objects (natural earth admin for example) is 3%-5% slower.
2022-12-05 15:07:32 -08:00

355 lines
7.9 KiB
C

/**********************************************************************
*
* PostGIS - Spatial Types for PostgreSQL
* http://postgis.net
*
* PostGIS is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* PostGIS is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with PostGIS. If not, see <http://www.gnu.org/licenses/>.
*
**********************************************************************
*
* Copyright 2002 Thamer Alharbash
* Copyright 2009 Paul Ramsey <pramsey@cleverelephant.ca>
*
**********************************************************************/
#include "liblwgeom_internal.h"
#include "stringbuffer.h"
/**
* Allocate a new stringbuffer_t. Use stringbuffer_destroy to free.
*/
stringbuffer_t*
stringbuffer_create(void)
{
return stringbuffer_create_with_size(STRINGBUFFER_STARTSIZE);
}
static void
stringbuffer_init_with_size(stringbuffer_t *s, size_t size)
{
s->str_start = lwalloc(size);
s->str_end = s->str_start;
s->capacity = size;
memset(s->str_start, 0, size);
}
void
stringbuffer_release(stringbuffer_t *s)
{
if ( s->str_start ) lwfree(s->str_start);
}
void
stringbuffer_init(stringbuffer_t *s)
{
stringbuffer_init_with_size(s, STRINGBUFFER_STARTSIZE);
}
void
stringbuffer_init_varlena(stringbuffer_t *s)
{
stringbuffer_init_with_size(s, STRINGBUFFER_STARTSIZE + LWVARHDRSZ);
/* Zero out LWVARHDRSZ bytes at the front of the buffer */
stringbuffer_append_len(s, "\0\0\0\0\0\0\0\0", LWVARHDRSZ);
}
/**
* Allocate a new stringbuffer_t. Use stringbuffer_destroy to free.
*/
stringbuffer_t*
stringbuffer_create_with_size(size_t size)
{
stringbuffer_t *s;
s = lwalloc(sizeof(stringbuffer_t));
stringbuffer_init_with_size(s, size);
return s;
}
/**
* Free the stringbuffer_t and all memory managed within it.
*/
void
stringbuffer_destroy(stringbuffer_t *s)
{
stringbuffer_release(s);
if ( s ) lwfree(s);
}
/**
* Reset the stringbuffer_t. Useful for starting a fresh string
* without the expense of freeing and re-allocating a new
* stringbuffer_t.
*/
void
stringbuffer_clear(stringbuffer_t *s)
{
s->str_start[0] = '\0';
s->str_end = s->str_start;
}
/**
* Return the last character in the buffer.
*/
char
stringbuffer_lastchar(stringbuffer_t *s)
{
if( s->str_end == s->str_start )
return 0;
return *(s->str_end - 1);
}
/**
* Returns a reference to the internal string being managed by
* the stringbuffer. The current string will be null-terminated
* within the internal string.
*/
const char*
stringbuffer_getstring(stringbuffer_t *s)
{
return s->str_start;
}
/**
* Returns a newly allocated string large enough to contain the
* current state of the string. Caller is responsible for
* freeing the return value.
*/
char*
stringbuffer_getstringcopy(stringbuffer_t *s)
{
size_t size = (s->str_end - s->str_start) + 1;
char *str = lwalloc(size);
memcpy(str, s->str_start, size);
str[size - 1] = '\0';
return str;
}
lwvarlena_t *
stringbuffer_getvarlena(stringbuffer_t *s)
{
lwvarlena_t *output = (lwvarlena_t *)(s->str_start);
LWSIZE_SET(output->size, (s->str_end - s->str_start));
return output;
}
lwvarlena_t *
stringbuffer_getvarlenacopy(stringbuffer_t *s)
{
size_t size = (s->str_end - s->str_start);
lwvarlena_t *output = (lwvarlena_t *)lwalloc(size + LWVARHDRSZ);
LWSIZE_SET(output->size, size + LWVARHDRSZ);
memcpy(output->data, s->str_start, size);
return output;
}
/**
* Returns the length of the current string, not including the
* null terminator (same behavior as strlen()).
*/
int
stringbuffer_getlength(stringbuffer_t *s)
{
return (s->str_end - s->str_start);
}
/**
* Clear the stringbuffer_t and re-start it with the specified string.
*/
void
stringbuffer_set(stringbuffer_t *s, const char *str)
{
stringbuffer_clear(s);
stringbuffer_append(s, str);
}
/**
* Copy the contents of src into dst.
*/
void
stringbuffer_copy(stringbuffer_t *dst, stringbuffer_t *src)
{
stringbuffer_set(dst, stringbuffer_getstring(src));
}
/**
* Appends a formatted string to the current string buffer,
* using the format and argument list provided. Returns -1 on error,
* check errno for reasons, documented in the printf man page.
*/
static int
stringbuffer_avprintf(stringbuffer_t *s, const char *fmt, va_list ap)
{
int maxlen = (s->capacity - (s->str_end - s->str_start));
int len = 0; /* Length of the output */
va_list ap2;
/* Make a copy of the variadic arguments, in case we need to print twice */
/* Print to our buffer */
va_copy(ap2, ap);
len = vsnprintf(s->str_end, maxlen, fmt, ap2);
va_end(ap2);
/* Propogate errors up */
if ( len < 0 )
#if defined(__MINGW64_VERSION_MAJOR)
va_copy(ap2, ap);
len = _vscprintf(fmt, ap2);/**Assume windows flaky vsnprintf that returns -1 if initial buffer to small and add more space **/
va_end(ap2);
#else
return len;
#endif
/* We didn't have enough space! */
/* Either Unix vsnprint returned write length larger than our buffer */
/* or Windows vsnprintf returned an error code. */
if ( len >= maxlen )
{
stringbuffer_makeroom(s, len + 1);
maxlen = (s->capacity - (s->str_end - s->str_start));
/* Try to print a second time */
len = vsnprintf(s->str_end, maxlen, fmt, ap);
/* Printing error? Error! */
if ( len < 0 ) return len;
/* Too long still? Error! */
if ( len >= maxlen ) return -1;
}
/* Move end pointer forward and return. */
s->str_end += len;
return len;
}
/**
* Appends a formatted string to the current string buffer,
* using the format and argument list provided.
* Returns -1 on error, check errno for reasons,
* as documented in the printf man page.
*/
int
stringbuffer_aprintf(stringbuffer_t *s, const char *fmt, ...)
{
int r;
va_list ap;
va_start(ap, fmt);
r = stringbuffer_avprintf(s, fmt, ap);
va_end(ap);
return r;
}
/**
* Trims whitespace off the end of the stringbuffer. Returns
* the number of characters trimmed.
*/
int
stringbuffer_trim_trailing_white(stringbuffer_t *s)
{
char *ptr = s->str_end;
int dist = 0;
/* Roll backwards until we hit a non-space. */
while( ptr > s->str_start )
{
ptr--;
if( (*ptr == ' ') || (*ptr == '\t') )
{
continue;
}
else
{
ptr++;
dist = s->str_end - ptr;
*ptr = '\0';
s->str_end = ptr;
return dist;
}
}
return dist;
}
/**
* Trims zeroes off the end of the last number in the stringbuffer.
* The number has to be the very last thing in the buffer. Only the
* last number will be trimmed. Returns the number of characters
* trimmed.
*
* eg: 1.22000 -> 1.22
* 1.0 -> 1
* 0.0 -> 0
*/
int
stringbuffer_trim_trailing_zeroes(stringbuffer_t *s)
{
char *ptr = s->str_end;
char *decimal_ptr = NULL;
int dist;
if ( s->str_end - s->str_start < 2)
return 0;
/* Roll backwards to find the decimal for this number */
while( ptr > s->str_start )
{
ptr--;
if ( *ptr == '.' )
{
decimal_ptr = ptr;
break;
}
if ( (*ptr >= '0') && (*ptr <= '9' ) )
continue;
else
break;
}
/* No decimal? Nothing to trim! */
if ( ! decimal_ptr )
return 0;
ptr = s->str_end;
/* Roll backwards again, with the decimal as stop point, trimming contiguous zeroes */
while( ptr >= decimal_ptr )
{
ptr--;
if ( *ptr == '0' )
continue;
else
break;
}
/* Huh, we get anywhere. Must not have trimmed anything. */
if ( ptr == s->str_end )
return 0;
/* If we stopped at the decimal, we want to null that out.
It we stopped on a numeral, we want to preserve that, so push the
pointer forward one space. */
if ( *ptr != '.' )
ptr++;
/* Add null terminator re-set the end of the stringbuffer. */
*ptr = '\0';
dist = s->str_end - ptr;
s->str_end = ptr;
return dist;
}