spice/server/cache-item.tmpl.cpp
Frediano Ziglio dd9b78fd78 Use smart pointers for RedPipeItem
Reduces usage of manual reference counting.
Avoids usage of new and use instead red::make_shared to both
create and own pipe items.
Use move semantic to reduce useless copies.

Signed-off-by: Frediano Ziglio <freddy77@gmail.com>
Acked-by: Julien Ropé <jrope@gmail.com>
2020-07-13 19:48:08 +01:00

141 lines
4.4 KiB
C++

/*
Copyright (C) 2009 Red Hat, Inc.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#include <config.h>
#if defined(CLIENT_CURSOR_CACHE)
#define CACHE_NAME cursor_cache
#define CACHE_HASH_KEY CURSOR_CACHE_HASH_KEY
#define CACHE_HASH_SIZE CURSOR_CACHE_HASH_SIZE
#define FUNC_NAME(name) red_cursor_cache_##name
#define VAR_NAME(name) cursor_cache_##name
#define CHANNELCLIENT CursorChannelClient
#elif defined(CLIENT_PALETTE_CACHE)
#define CACHE_NAME palette_cache
#define CACHE_HASH_KEY PALETTE_CACHE_HASH_KEY
#define CACHE_HASH_SIZE PALETTE_CACHE_HASH_SIZE
#define FUNC_NAME(name) red_palette_cache_##name
#define VAR_NAME(name) palette_cache_##name
#define CHANNELCLIENT DisplayChannelClient
#else
#error "no cache type."
#endif
static RedCacheItem *FUNC_NAME(find)(CHANNELCLIENT *channel_client, uint64_t id)
{
RedCacheItem *item = channel_client->priv->CACHE_NAME[CACHE_HASH_KEY(id)];
while (item) {
if (item->id == id) {
ring_remove(&item->lru_link);
ring_add(&channel_client->priv->VAR_NAME(lru), &item->lru_link);
break;
}
item = item->next;
}
return item;
}
static void FUNC_NAME(remove)(CHANNELCLIENT *channel_client, RedCacheItem *item)
{
RedCacheItem **now;
spice_assert(item);
now = &channel_client->priv->CACHE_NAME[CACHE_HASH_KEY(item->id)];
for (;;) {
spice_assert(*now);
if (*now == item) {
*now = item->next;
break;
}
now = &(*now)->next;
}
ring_remove(&item->lru_link);
channel_client->priv->VAR_NAME(available) += item->size;
// see "Optimization" comment on add function below
auto id = item->id;
RedCachePipeItem *pipe_item = reinterpret_cast<RedCachePipeItem*>(item);
new (pipe_item) RedCachePipeItem();
pipe_item->inval_one.id = id;
channel_client->pipe_add_tail(RedPipeItemPtr(pipe_item)); // for now
}
static int FUNC_NAME(add)(CHANNELCLIENT *channel_client, uint64_t id, size_t size)
{
RedCacheItem *item;
int key;
/* Optimization: allocate memory in order to be able to store
* both cache item and pipe item to be able to reuse it when
* we need to remove cache telling client */
union RedCachePoolItem {
RedCacheItem cache_item;
RedCachePipeItem pipe_item;
};
item = (RedCacheItem *) g_new(RedCachePoolItem, 1);
channel_client->priv->VAR_NAME(available) -= size;
SPICE_VERIFY(SPICE_OFFSETOF(RedCacheItem, lru_link) == 0);
while (channel_client->priv->VAR_NAME(available) < 0) {
RedCacheItem *tail = SPICE_CONTAINEROF(ring_get_tail(&channel_client->priv->VAR_NAME(lru)),
RedCacheItem, lru_link);
if (!tail) {
channel_client->priv->VAR_NAME(available) += size;
g_free(item);
return FALSE;
}
FUNC_NAME(remove)(channel_client, tail);
}
item->next = channel_client->priv->CACHE_NAME[(key = CACHE_HASH_KEY(id))];
channel_client->priv->CACHE_NAME[key] = item;
ring_item_init(&item->lru_link);
ring_add(&channel_client->priv->VAR_NAME(lru), &item->lru_link);
item->id = id;
item->size = size;
return TRUE;
}
static void FUNC_NAME(reset)(CHANNELCLIENT *channel_client, long size)
{
int i;
for (i = 0; i < CACHE_HASH_SIZE; i++) {
while (channel_client->priv->CACHE_NAME[i]) {
RedCacheItem *item = channel_client->priv->CACHE_NAME[i];
channel_client->priv->CACHE_NAME[i] = item->next;
g_free(item);
}
}
ring_init(&channel_client->priv->VAR_NAME(lru));
channel_client->priv->VAR_NAME(available) = size;
}
#undef CACHE_NAME
#undef CACHE_HASH_KEY
#undef CACHE_HASH_SIZE
#undef FUNC_NAME
#undef VAR_NAME
#undef CHANNELCLIENT