]> git.sesse.net Git - vlc/commitdiff
* ./src/misc/variables.c, ./include/variables.h: a few functions for
authorSam Hocevar <sam@videolan.org>
Fri, 11 Oct 2002 11:05:52 +0000 (11:05 +0000)
committerSam Hocevar <sam@videolan.org>
Fri, 11 Oct 2002 11:05:52 +0000 (11:05 +0000)
    in-object variable storage. Unused yet (not even compiled).

include/variables.h [new file with mode: 0644]
src/misc/variables.c [new file with mode: 0644]

diff --git a/include/variables.h b/include/variables.h
new file mode 100644 (file)
index 0000000..3496573
--- /dev/null
@@ -0,0 +1,71 @@
+/*****************************************************************************
+ * variables.h: variables handling
+ *****************************************************************************
+ * Copyright (C) 2002 VideoLAN
+ * $Id: variables.h,v 1.1 2002/10/11 11:05:52 sam Exp $
+ *
+ * Authors: Samuel Hocevar <sam@zoy.org>
+ *
+ * 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 Place - Suite 330, Boston, MA  02111, USA.
+ *****************************************************************************/
+
+/* Variable types - probably very incomplete */
+#define VLC_VAR_BOOL      0x0100
+#define VLC_VAR_INTEGER   0x0200
+#define VLC_VAR_STRING    0x0300
+#define VLC_VAR_MODULE    0x0301
+#define VLC_VAR_FILE      0x0302
+#define VLC_VAR_FLOAT     0x0400
+#define VLC_VAR_TIME      0x0500
+#define VLC_VAR_ADDRESS   0x0600
+
+/*****************************************************************************
+ * vlc_value_t is the common union for variable values; variable_t is the
+ * structure describing a variable. 
+ *****************************************************************************/
+struct variable_t
+{
+    u32          i_hash;
+    char *       psz_name;
+
+    int          i_type;
+    vlc_value_t  val;
+
+    /* Lots of other things that can be added */
+    vlc_bool_t   b_set;
+    vlc_bool_t   b_active;
+};
+
+/*****************************************************************************
+ * Prototypes
+ *****************************************************************************/
+VLC_EXPORT( void, __var_Create, ( vlc_object_t *, const char *, int ) );
+VLC_EXPORT( void, __var_Destroy, ( vlc_object_t *, const char * ) );
+
+VLC_EXPORT( int, __var_Set, ( vlc_object_t *, const char *, vlc_value_t ) );
+VLC_EXPORT( int, __var_Get, ( vlc_object_t *, const char *, vlc_value_t * ) );
+
+#define var_Create(a,b,c) \
+    __var_Create( VLC_OBJECT(a), b, c )
+
+#define var_Destroy(a,b) \
+    __var_Destroy( VLC_OBJECT(a), b )
+
+#define var_Set(a,b,c) \
+    __var_Set( VLC_OBJECT(a), b, c )
+
+#define var_Get(a,b,c) \
+    __var_Get( VLC_OBJECT(a), b, c )
+
diff --git a/src/misc/variables.c b/src/misc/variables.c
new file mode 100644 (file)
index 0000000..bcdc96b
--- /dev/null
@@ -0,0 +1,391 @@
+/*****************************************************************************
+ * variables.c: routines for object variables handling
+ *****************************************************************************
+ * Copyright (C) 2002 VideoLAN
+ * $Id: variables.c,v 1.1 2002/10/11 11:05:52 sam Exp $
+ *
+ * Authors: Samuel Hocevar <sam@zoy.org>
+ *
+ * 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 Place - Suite 330, Boston, MA  02111, USA.
+ *****************************************************************************/
+
+/*****************************************************************************
+ * Preamble
+ *****************************************************************************/
+#include <vlc/vlc.h>
+
+#ifdef HAVE_STDLIB_H
+#   include <stdlib.h>                                          /* realloc() */
+#endif
+
+/*****************************************************************************
+ * Local prototypes
+ *****************************************************************************/
+static u32 HashString     ( const char * );
+static int Insert         ( variable_t *, int, const char * );
+static int InsertInner    ( variable_t *, int, u32 );
+static int Lookup         ( variable_t *, int, const char * );
+static int LookupInner    ( variable_t *, int, u32 );
+
+/*****************************************************************************
+ * var_Create: initialize a vlc variable
+ *****************************************************************************
+ * We hash the given string and insert it into the sorted list. The insertion
+ * may require slow memory copies, but think about what we gain in the log(n)
+ * lookup phase when setting/getting the variable value!
+ *****************************************************************************/
+void __var_Create( vlc_object_t *p_this, const char *psz_name, int i_type )
+{
+    int i_new;
+
+    vlc_mutex_lock( &p_this->var_lock );
+
+    if( (p_this->i_vars & 15) == 15 )
+    {
+        p_this->p_vars = realloc( p_this->p_vars,
+                                  (p_this->i_vars+17) * sizeof(variable_t) );
+    }
+
+    i_new = Insert( p_this->p_vars, p_this->i_vars, psz_name );
+
+    memmove( p_this->p_vars + i_new + 1,
+             p_this->p_vars + i_new,
+             (p_this->i_vars - i_new) * sizeof(variable_t) );
+
+    p_this->p_vars[i_new].i_hash = HashString( psz_name );
+    p_this->p_vars[i_new].psz_name = strdup( psz_name );
+
+    p_this->p_vars[i_new].i_type = i_type;
+    memset( &p_this->p_vars[i_new].val, 0, sizeof(vlc_value_t) );
+
+    p_this->p_vars[i_new].b_set = VLC_FALSE;
+    p_this->p_vars[i_new].b_active = VLC_TRUE;
+
+    p_this->i_vars++;
+
+    vlc_mutex_unlock( &p_this->var_lock );
+}
+
+/*****************************************************************************
+ * var_Destroy: destroy a vlc variable
+ *****************************************************************************
+ * Look for the variable and destroy it if it is found. As in var_Create we
+ * do a call to memmove() but we have performance counterparts elsewhere.
+ *****************************************************************************/
+void __var_Destroy( vlc_object_t *p_this, const char *psz_name )
+{
+    int i_del;
+
+    vlc_mutex_lock( &p_this->var_lock );
+
+    i_del = Lookup( p_this->p_vars, p_this->i_vars, psz_name );
+
+    if( i_del < 0 )
+    {
+        msg_Err( p_this, "variable %s was not found", psz_name );
+        vlc_mutex_unlock( &p_this->var_lock );
+        return;
+    }
+
+    /* Free value if needed */
+    switch( p_this->p_vars[i_del].i_type )
+    {
+        case VLC_VAR_STRING:
+        case VLC_VAR_MODULE:
+        case VLC_VAR_FILE:
+            if( p_this->p_vars[i_del].b_set
+                 && p_this->p_vars[i_del].val.psz_string )
+            {
+                free( p_this->p_vars[i_del].val.psz_string );
+            }
+            break;
+    }
+
+    free( p_this->p_vars[i_del].psz_name );
+
+    memmove( p_this->p_vars + i_del,
+             p_this->p_vars + i_del + 1,
+             (p_this->i_vars - i_del - 1) * sizeof(variable_t) );
+
+    if( (p_this->i_vars & 15) == 0 )
+    {
+        p_this->p_vars = realloc( p_this->p_vars,
+                          (p_this->i_vars) * sizeof( variable_t ) );
+    }
+
+    p_this->i_vars--;
+
+    vlc_mutex_unlock( &p_this->var_lock );
+}
+
+/*****************************************************************************
+ * var_Set: set a variable's value
+ *****************************************************************************
+ *
+ *****************************************************************************/
+int __var_Set( vlc_object_t *p_this, const char *psz_name, vlc_value_t val )
+{
+    int i_var;
+
+    vlc_mutex_lock( &p_this->var_lock );
+
+    i_var = Lookup( p_this->p_vars, p_this->i_vars, psz_name );
+
+    if( i_var < 0 )
+    {
+        msg_Err( p_this, "variable %s was not found", psz_name );
+        vlc_mutex_unlock( &p_this->var_lock );
+        return VLC_EVAR;
+    }
+
+    /* Duplicate value if needed */
+    switch( p_this->p_vars[i_var].i_type )
+    {
+        case VLC_VAR_STRING:
+        case VLC_VAR_MODULE:
+        case VLC_VAR_FILE:
+            if( p_this->p_vars[i_var].b_set
+                 && p_this->p_vars[i_var].val.psz_string )
+            {
+                free( p_this->p_vars[i_var].val.psz_string );
+            }
+            if( val.psz_string )
+            {
+                val.psz_string = strdup( val.psz_string );
+            }
+            break;
+    }
+
+    p_this->p_vars[i_var].val = val;
+    p_this->p_vars[i_var].b_set = VLC_TRUE;
+
+    /* XXX: callback stuff will go here */
+
+    vlc_mutex_unlock( &p_this->var_lock );
+
+    return VLC_SUCCESS;
+}
+
+/*****************************************************************************
+ * var_Get: get a variable's value
+ *****************************************************************************
+ *
+ *****************************************************************************/
+int __var_Get( vlc_object_t *p_this, const char *psz_name,
+                                     vlc_value_t *p_value )
+{
+    int i_var;
+
+    vlc_mutex_lock( &p_this->var_lock );
+
+    i_var = Lookup( p_this->p_vars, p_this->i_vars, psz_name );
+
+    if( i_var < 0 )
+    {
+        msg_Err( p_this, "variable %s was not found", psz_name );
+        vlc_mutex_unlock( &p_this->var_lock );
+        return VLC_EVAR;
+    }
+
+    if( !p_this->p_vars[i_var].b_set )
+    {
+        msg_Err( p_this, "variable %s is not set", psz_name );
+        vlc_mutex_unlock( &p_this->var_lock );
+        return VLC_EVAR;
+    }
+
+    *p_value = p_this->p_vars[i_var].val;
+
+    /* Duplicate value if needed */
+    switch( p_this->p_vars[i_var].i_type )
+    {
+        case VLC_VAR_STRING:
+        case VLC_VAR_MODULE:
+        case VLC_VAR_FILE:
+            if( p_value->psz_string )
+            {
+                p_value->psz_string = strdup( p_value->psz_string );
+            }
+            break;
+    }
+
+    vlc_mutex_unlock( &p_this->var_lock );
+
+    return VLC_SUCCESS;
+}
+
+/* Following functions are local */
+
+/*****************************************************************************
+ * HashString: our cool hash function
+ *****************************************************************************
+ * This function is not intended to be crypto-secure, we only want it to be
+ * fast and not suck too much. This one is pretty fast and did 0 collisions
+ * in wenglish's dictionary.
+ *****************************************************************************/
+static u32 HashString( const char *psz_string )
+{
+    u32 i_hash = 0;
+
+    while( *psz_string )
+    {
+        i_hash += *psz_string++;
+        i_hash += i_hash << 10;
+        i_hash ^= i_hash >> 8;
+    }
+
+    return i_hash;
+}
+
+/*****************************************************************************
+ * Insert: find an empty slot to insert a new variable
+ *****************************************************************************
+ * We use a recursive inner function indexed on the hash. This function does
+ * nothing in the rare cases where a collision may occur, see Lookup()
+ * to see how we handle them.
+ * XXX: does this really need to be written recursively?
+ *****************************************************************************/
+static int Insert( variable_t *p_vars, int i_count, const char *psz_name )
+{
+    if( i_count == 0 )
+    {
+        return 0;
+    }
+
+    return InsertInner( p_vars, i_count, HashString( psz_name ) );
+}
+
+static int InsertInner( variable_t *p_vars, int i_count, u32 i_hash )
+{
+    int i_middle;
+
+    if( i_hash <= p_vars[0].i_hash )
+    {
+        return 0;
+    }
+
+    if( i_hash >= p_vars[i_count - 1].i_hash )
+    {
+        return i_count;
+    }
+
+    i_middle = i_count / 2;
+
+    /* We know that 0 < i_middle */
+    if( i_hash < p_vars[i_middle].i_hash )
+    {
+        return InsertInner( p_vars, i_middle, i_hash );
+    }
+
+    /* We know that i_middle + 1 < i_count */
+    if( i_hash > p_vars[i_middle + 1].i_hash )
+    {
+        return i_middle + 1 + InsertInner( p_vars + i_middle + 1,
+                                           i_count - i_middle - 1,
+                                           i_hash );
+    }
+
+    return i_middle + 1;
+}
+
+/*****************************************************************************
+ * Lookup: find an existing variable given its name
+ *****************************************************************************
+ * We use a recursive inner function indexed on the hash. Care is taken of
+ * possible hash collisions.
+ * XXX: does this really need to be written recursively?
+ *****************************************************************************/
+static int Lookup( variable_t *p_vars, int i_count, const char *psz_name )
+{
+    u32 i_hash;
+    int i, i_pos;
+
+    if( i_count == 0 )
+    {
+        return -1;
+    }
+
+    i_hash = HashString( psz_name );
+
+    i_pos = LookupInner( p_vars, i_count, i_hash );
+
+    /* Hash not found */
+    if( i_hash != p_vars[i_pos].i_hash )
+    {
+        return -1;
+    }
+
+    /* Hash found, entry found */
+    if( !strcmp( psz_name, p_vars[i_pos].psz_name ) )
+    {
+        return i_pos;
+    }
+
+    /* Hash collision! This should be very rare, but we cannot guarantee
+     * it will never happen. Just do an exhaustive search amongst all
+     * entries with the same hash. */
+    for( i = i_pos - 1 ; i > 0 && i_hash == p_vars[i].i_hash ; i-- )
+    {
+        if( !strcmp( psz_name, p_vars[i].psz_name ) )
+        {
+            return i;
+        }
+    }
+
+    for( i = i_pos + 1 ; i < i_count && i_hash == p_vars[i].i_hash ; i++ )
+    {
+        if( !strcmp( psz_name, p_vars[i].psz_name ) )
+        {
+            return i;
+        }
+    }
+
+    /* Hash found, but entry not found */
+    return -1;
+}
+
+static int LookupInner( variable_t *p_vars, int i_count, u32 i_hash )
+{
+    int i_middle;
+
+    if( i_hash <= p_vars[0].i_hash )
+    {
+        return 0;
+    }
+
+    if( i_hash >= p_vars[i_count-1].i_hash )
+    {
+        return i_count - 1;
+    }
+
+    i_middle = i_count / 2;
+
+    /* We know that 0 < i_middle */
+    if( i_hash < p_vars[i_middle].i_hash )
+    {
+        return LookupInner( p_vars, i_middle, i_hash );
+    }
+
+    /* We know that i_middle + 1 < i_count */
+    if( i_hash > p_vars[i_middle].i_hash )
+    {
+        return i_middle + LookupInner( p_vars + i_middle,
+                                       i_count - i_middle,
+                                       i_hash );
+    }
+
+    return i_middle;
+}
+