2004-11-12 Not Zed <NotZed@Ximian.com> ** Merge in notzed-messageinfo-branch, fix some minor conflicts. svn path=/trunk/; revision=27898
222 lines
4.6 KiB
C
222 lines
4.6 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
|
|
/*
|
|
* Authors: Jeffrey Stedfast <fejj@ximian.com>
|
|
*
|
|
* 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 <config.h>
|
|
#endif
|
|
|
|
#include <string.h>
|
|
#include <pthread.h>
|
|
|
|
#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);
|
|
}
|