Blame src/xcb_ext.c

Packit 071ada
/* Copyright (C) 2001-2004 Bart Massey and Jamey Sharp.
Packit 071ada
 *
Packit 071ada
 * Permission is hereby granted, free of charge, to any person obtaining a
Packit 071ada
 * copy of this software and associated documentation files (the "Software"),
Packit 071ada
 * to deal in the Software without restriction, including without limitation
Packit 071ada
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
Packit 071ada
 * and/or sell copies of the Software, and to permit persons to whom the
Packit 071ada
 * Software is furnished to do so, subject to the following conditions:
Packit 071ada
 * 
Packit 071ada
 * The above copyright notice and this permission notice shall be included in
Packit 071ada
 * all copies or substantial portions of the Software.
Packit 071ada
 * 
Packit 071ada
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit 071ada
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit 071ada
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit 071ada
 * AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
Packit 071ada
 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
Packit 071ada
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Packit 071ada
 * 
Packit 071ada
 * Except as contained in this notice, the names of the authors or their
Packit 071ada
 * institutions shall not be used in advertising or otherwise to promote the
Packit 071ada
 * sale, use or other dealings in this Software without prior written
Packit 071ada
 * authorization from the authors.
Packit 071ada
 */
Packit 071ada
Packit 071ada
/* A cache for QueryExtension results. */
Packit 071ada
Packit 071ada
#ifdef HAVE_CONFIG_H
Packit 071ada
#include "config.h"
Packit 071ada
#endif
Packit 071ada
Packit 071ada
#include <stdlib.h>
Packit 071ada
#include <string.h>
Packit 071ada
Packit 071ada
#include "xcb.h"
Packit 071ada
#include "xcbext.h"
Packit 071ada
#include "xcbint.h"
Packit 071ada
Packit 071ada
typedef struct lazyreply {
Packit 071ada
    enum lazy_reply_tag tag;
Packit 071ada
    union {
Packit 071ada
        xcb_query_extension_cookie_t cookie;
Packit 071ada
        xcb_query_extension_reply_t *reply;
Packit 071ada
    } value;
Packit 071ada
} lazyreply;
Packit 071ada
Packit 071ada
static lazyreply *get_index(xcb_connection_t *c, int idx)
Packit 071ada
{
Packit 071ada
    if(idx > c->ext.extensions_size)
Packit 071ada
    {
Packit 071ada
        int new_size = idx << 1;
Packit 071ada
        lazyreply *new_extensions = realloc(c->ext.extensions, sizeof(lazyreply) * new_size);
Packit 071ada
        if(!new_extensions)
Packit 071ada
            return 0;
Packit 071ada
        memset(new_extensions + c->ext.extensions_size, 0, sizeof(lazyreply) * (new_size - c->ext.extensions_size));
Packit 071ada
        c->ext.extensions = new_extensions;
Packit 071ada
        c->ext.extensions_size = new_size;
Packit 071ada
    }
Packit 071ada
    return c->ext.extensions + idx - 1;
Packit 071ada
}
Packit 071ada
Packit 071ada
static lazyreply *get_lazyreply(xcb_connection_t *c, xcb_extension_t *ext)
Packit 071ada
{
Packit 071ada
    static pthread_mutex_t global_lock = PTHREAD_MUTEX_INITIALIZER;
Packit 071ada
    static int next_global_id;
Packit 071ada
Packit 071ada
    lazyreply *data;
Packit 071ada
Packit 071ada
    pthread_mutex_lock(&global_lock);
Packit 071ada
    if(!ext->global_id)
Packit 071ada
        ext->global_id = ++next_global_id;
Packit 071ada
    pthread_mutex_unlock(&global_lock);
Packit 071ada
Packit 071ada
    data = get_index(c, ext->global_id);
Packit 071ada
    if(data && data->tag == LAZY_NONE)
Packit 071ada
    {
Packit 071ada
        /* cache miss: query the server */
Packit 071ada
        data->tag = LAZY_COOKIE;
Packit 071ada
        data->value.cookie = xcb_query_extension(c, strlen(ext->name), ext->name);
Packit 071ada
    }
Packit 071ada
    return data;
Packit 071ada
}
Packit 071ada
Packit 071ada
/* Public interface */
Packit 071ada
Packit 071ada
/* Do not free the returned xcb_query_extension_reply_t - on return, it's aliased
Packit 071ada
 * from the cache. */
Packit 071ada
const xcb_query_extension_reply_t *xcb_get_extension_data(xcb_connection_t *c, xcb_extension_t *ext)
Packit 071ada
{
Packit 071ada
    lazyreply *data;
Packit 071ada
    if(c->has_error)
Packit 071ada
        return 0;
Packit 071ada
Packit 071ada
    pthread_mutex_lock(&c->ext.lock);
Packit 071ada
    data = get_lazyreply(c, ext);
Packit 071ada
    if(data && data->tag == LAZY_COOKIE)
Packit 071ada
    {
Packit 071ada
        data->tag = LAZY_FORCED;
Packit 071ada
        data->value.reply = xcb_query_extension_reply(c, data->value.cookie, 0);
Packit 071ada
    }
Packit 071ada
    pthread_mutex_unlock(&c->ext.lock);
Packit 071ada
Packit 071ada
    return data ? data->value.reply : 0;
Packit 071ada
}
Packit 071ada
Packit 071ada
void xcb_prefetch_extension_data(xcb_connection_t *c, xcb_extension_t *ext)
Packit 071ada
{
Packit 071ada
    if(c->has_error)
Packit 071ada
        return;
Packit 071ada
    pthread_mutex_lock(&c->ext.lock);
Packit 071ada
    get_lazyreply(c, ext);
Packit 071ada
    pthread_mutex_unlock(&c->ext.lock);
Packit 071ada
}
Packit 071ada
Packit 071ada
/* Private interface */
Packit 071ada
Packit 071ada
int _xcb_ext_init(xcb_connection_t *c)
Packit 071ada
{
Packit 071ada
    if(pthread_mutex_init(&c->ext.lock, 0))
Packit 071ada
        return 0;
Packit 071ada
    return 1;
Packit 071ada
}
Packit 071ada
Packit 071ada
void _xcb_ext_destroy(xcb_connection_t *c)
Packit 071ada
{
Packit 071ada
    pthread_mutex_destroy(&c->ext.lock);
Packit 071ada
    while(c->ext.extensions_size-- > 0)
Packit 071ada
        if(c->ext.extensions[c->ext.extensions_size].tag == LAZY_FORCED)
Packit 071ada
            free(c->ext.extensions[c->ext.extensions_size].value.reply);
Packit 071ada
    free(c->ext.extensions);
Packit 071ada
}