| /*************************************************************************** |
| * __________ __ ___. |
| * Open \______ \ ____ ____ | | _\_ |__ _______ ___ |
| * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / |
| * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < |
| * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ |
| * \/ \/ \/ \/ \/ |
| * |
| * Copyright (C) 2003 Tat Tang |
| * |
| * 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY |
| * KIND, either express or implied. |
| * |
| ****************************************************************************/ |
| |
| #include <string.h> |
| #include "font_cache.h" |
| #include "debug.h" |
| |
| /******************************************************************************* |
| * font_cache_lru_init |
| ******************************************************************************/ |
| static void font_cache_lru_init(void* data) |
| { |
| struct font_cache_entry* p = data; |
| p->_char_code = 0xffff; /* assume invalid char */ |
| } |
| |
| /******************************************************************************* |
| * font_cache_create |
| ******************************************************************************/ |
| void font_cache_create( |
| struct font_cache* fcache, |
| void *buf, |
| int buf_size, |
| int bitmap_bytes_size) |
| { |
| int font_cache_entry_size = |
| sizeof(struct font_cache_entry) + bitmap_bytes_size; |
| |
| /* make sure font cache entries are a multiple of 16 bits */ |
| if (font_cache_entry_size % 2 != 0) |
| font_cache_entry_size++; |
| |
| int cache_size = buf_size / |
| (font_cache_entry_size + LRU_SLOT_OVERHEAD + sizeof(short)); |
| |
| fcache->_size = 1; |
| fcache->_capacity = cache_size; |
| |
| /* set up index */ |
| fcache->_index = buf; |
| |
| /* set up lru list */ |
| unsigned char* lru_buf = buf; |
| lru_buf += sizeof(short) * cache_size; |
| lru_create(&fcache->_lru, lru_buf, cache_size, font_cache_entry_size); |
| |
| /* initialise cache */ |
| lru_traverse(&fcache->_lru, font_cache_lru_init); |
| short i; |
| for (i = 0; i < cache_size; i++) |
| fcache->_index[i] = i; /* small cheat here */ |
| } |
| |
| /******************************************************************************* |
| * font_cache_index_of |
| ******************************************************************************/ |
| static int font_cache_index_of( |
| struct font_cache* fcache, |
| unsigned short char_code) |
| { |
| struct font_cache_entry* p; |
| int left, right, mid, c; |
| left = 0; |
| right = fcache->_size - 1; |
| |
| do |
| { |
| mid = (left + right) / 2; |
| |
| p = lru_data(&fcache->_lru, fcache->_index[mid]); |
| c = p->_char_code - char_code; |
| |
| if (c == 0) |
| return mid; |
| |
| if (c < 0) |
| left = mid + 1; |
| else |
| right = mid - 1; |
| } |
| while (left <= right); |
| |
| return -1; |
| } |
| |
| /******************************************************************************* |
| * font_cache_insertion_point |
| ******************************************************************************/ |
| static int font_cache_insertion_point( |
| struct font_cache* fcache, |
| unsigned short char_code) |
| { |
| struct font_cache_entry* p; |
| short *index = fcache->_index; |
| |
| p = lru_data(&fcache->_lru, index[0]); |
| if (char_code < p->_char_code) |
| return -1; |
| |
| p = lru_data(&fcache->_lru, index[fcache->_capacity - 1]); |
| if (char_code > p->_char_code) |
| return fcache->_capacity - 1; |
| |
| int left, right, mid, c; |
| |
| left = 0; |
| right = fcache->_capacity - 1; |
| |
| do |
| { |
| mid = (left + right) / 2; |
| |
| p = lru_data(&fcache->_lru, index[mid]); |
| c = char_code - p->_char_code; |
| |
| if (c >= 0) |
| { |
| p = lru_data(&fcache->_lru, index[mid+1]); |
| int z = char_code - p->_char_code; |
| |
| if (z < 0) |
| return mid; |
| |
| if (z == 0) |
| return mid + 1; |
| } |
| |
| |
| if (c > 0) |
| left = mid + 1; |
| else |
| right = mid - 1; |
| } |
| while (left <= right); |
| |
| /* not found */ |
| return -2; |
| } |
| |
| /******************************************************************************* |
| * font_cache_get |
| ******************************************************************************/ |
| struct font_cache_entry* font_cache_get( |
| struct font_cache* fcache, |
| unsigned short char_code, |
| void (*callback) (struct font_cache_entry* p, void *callback_data), |
| void *callback_data) |
| { |
| int insertion_point = font_cache_insertion_point(fcache, char_code); |
| if (insertion_point >= 0) |
| { |
| short lru_handle = fcache->_index[insertion_point]; |
| struct font_cache_entry* p = lru_data(&fcache->_lru, lru_handle); |
| if (p->_char_code == char_code) |
| { |
| lru_touch(&fcache->_lru, lru_handle); |
| return lru_data(&fcache->_lru, lru_handle); |
| } |
| } |
| |
| /* not found */ |
| short lru_handle_to_replace = fcache->_lru._head; |
| struct font_cache_entry* p = |
| lru_data(&fcache->_lru, lru_handle_to_replace); |
| int index_to_replace = font_cache_index_of(fcache, p->_char_code); |
| |
| if (insertion_point < index_to_replace) |
| { |
| /* shift memory up */ |
| memmove(fcache->_index + insertion_point + 2, |
| fcache->_index + insertion_point + 1, |
| (index_to_replace - insertion_point - 1) * sizeof(short)); |
| |
| /* add to index */ |
| fcache->_index[insertion_point + 1] = lru_handle_to_replace; |
| } |
| else if (insertion_point > index_to_replace) |
| { |
| /* shift memory down */ |
| memmove(fcache->_index + index_to_replace, |
| fcache->_index + index_to_replace + 1, |
| (insertion_point - index_to_replace) * sizeof(short)); |
| |
| /* add to index */ |
| fcache->_index[insertion_point] = lru_handle_to_replace; |
| } |
| |
| /* load new entry into cache */ |
| lru_touch(&fcache->_lru, lru_handle_to_replace); |
| |
| if (fcache->_size < fcache->_capacity) |
| fcache->_size++; |
| |
| p->_char_code = char_code; |
| callback(p, callback_data); |
| |
| return p; |
| } |