/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Authors: Jeffrey Stedfast * * Copyright 2002 Ximian, Inc. (www.ximian.com) * * This program 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. * * This program 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 this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Street #330, Boston, MA 02111-1307, USA. * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include "camel-string-utils.h" int camel_strcase_equal (gconstpointer a, gconstpointer b) { return (g_ascii_strcasecmp ((const char *) a, (const char *) b) == 0); } guint camel_strcase_hash (gconstpointer v) { const char *p = (char *) v; guint h = 0, g; for ( ; *p != '\0'; p++) { h = (h << 4) + g_ascii_toupper (*p); if ((g = h & 0xf0000000)) { h = h ^ (g >> 24); h = h ^ g; } } return h; } static void free_string (gpointer string, gpointer user_data) { g_free (string); } void camel_string_list_free (GList *string_list) { if (string_list == NULL) return; g_list_foreach (string_list, free_string, NULL); g_list_free (string_list); } char * camel_strstrcase (const char *haystack, const char *needle) { /* find the needle in the haystack neglecting case */ const char *ptr; guint len; g_return_val_if_fail (haystack != NULL, NULL); g_return_val_if_fail (needle != NULL, NULL); len = strlen (needle); if (len > strlen (haystack)) return NULL; if (len == 0) return (char *) haystack; for (ptr = haystack; *(ptr + len - 1) != '\0'; ptr++) if (!strncasecmp (ptr, needle, len)) return (char *) ptr; return NULL; } const char * camel_strdown (char *str) { register char *s = str; while (*s) { if (*s >= 'A' && *s <= 'Z') *s += 0x20; s++; } return str; } /** * camel_tolower: * @c: * * ASCII to-lower function. * * Return value: **/ char camel_tolower(char c) { if (c >= 'A' && c <= 'Z') c |= 0x20; return c; } /** * camel_toupper: * @c: * * ASCII to-upper function. * * Return value: **/ char camel_toupper(char c) { if (c >= 'a' && c <= 'z') c &= ~0x20; return c; } /* working stuff for pstrings */ static pthread_mutex_t pstring_lock = PTHREAD_MUTEX_INITIALIZER; static GHashTable *pstring_table = NULL; /** * camel_pstring_strdup: * @s: String to copy. * * Create a new pooled string entry for the string @s. A pooled * string is a table where common strings are uniquified to the same * pointer value. They are also refcounted, so freed when no longer * in use. In a thread-safe manner. * * The NULL and empty strings are special cased to constant values. * * Return value: A pointer to an equivalent string of @s. Use * camel_pstring_free() when it is no longer needed. **/ const char *camel_pstring_strdup(const char *s) { char *p; void *pcount; int count; if (s == NULL) return NULL; if (s[0] == 0) return ""; pthread_mutex_lock(&pstring_lock); if (pstring_table == NULL) pstring_table = g_hash_table_new(g_str_hash, g_str_equal); if (g_hash_table_lookup_extended(pstring_table, s, (void **)&p, &pcount)) { count = GPOINTER_TO_INT(pcount)+1; g_hash_table_insert(pstring_table, p, GINT_TO_POINTER(count)); } else { p = g_strdup(s); g_hash_table_insert(pstring_table, p, GINT_TO_POINTER(1)); } pthread_mutex_unlock(&pstring_lock); return p; } /** * camel_pstring_free: * @s: String to free. * * De-ref a pooled string. If no more refs exist to this string, it will be deallocated. * * NULL and the empty string are special cased. **/ void camel_pstring_free(const char *s) { char *p; void *pcount; int count; if (pstring_table == NULL) return; if (s == NULL || s[0] == 0) return; pthread_mutex_lock(&pstring_lock); if (g_hash_table_lookup_extended(pstring_table, s, (void **)&p, &pcount)) { count = GPOINTER_TO_INT(pcount)-1; if (count == 0) { g_hash_table_remove(pstring_table, p); g_free(p); } else { g_hash_table_insert(pstring_table, p, GINT_TO_POINTER(count)); } } else { g_warning("Trying to free string not allocated from the pool '%s'", s); } pthread_mutex_unlock(&pstring_lock); }