mirror of
https://github.com/torvalds/linux.git
synced 2026-05-12 16:18:45 +02:00
Implement pitch and size calculation for a single font glyph in the new helpers font_glyph_pitch() and font_glyph_size(). Replace the instances where the calculations are open-coded. Note that in the case of fbcon console rotation, the parameters for a glyph's width and height might be reversed. This is intentional. v2: - fix typos in commit message Signed-off-by: Thomas Zimmermann <tzimmermann@suse.de> Signed-off-by: Helge Deller <deller@gmx.de>
388 lines
9.4 KiB
C
388 lines
9.4 KiB
C
/*
|
|
* `Soft' font definitions
|
|
*
|
|
* Created 1995 by Geert Uytterhoeven
|
|
* Rewritten 1998 by Martin Mares <mj@ucw.cz>
|
|
*
|
|
* 2001 - Documented with DocBook
|
|
* - Brad Douglas <brad@neruo.com>
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
|
* License. See the file COPYING in the main directory of this archive
|
|
* for more details.
|
|
*/
|
|
|
|
#include <linux/container_of.h>
|
|
#include <linux/kd.h>
|
|
#include <linux/module.h>
|
|
#include <linux/overflow.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/string.h>
|
|
#include <linux/types.h>
|
|
|
|
#if defined(__mc68000__)
|
|
#include <asm/setup.h>
|
|
#endif
|
|
|
|
#include "font.h"
|
|
|
|
#define console_font_pitch(font) font_glyph_pitch((font)->width)
|
|
|
|
/*
|
|
* Helpers for font_data_t
|
|
*/
|
|
|
|
/* Extra word getters */
|
|
#define REFCOUNT(fd) (((int *)(fd))[-1])
|
|
#define FNTSIZE(fd) (((int *)(fd))[-2])
|
|
#define FNTSUM(fd) (((int *)(fd))[-4])
|
|
|
|
static struct font_data *to_font_data_struct(font_data_t *fd)
|
|
{
|
|
return container_of(fd, struct font_data, data[0]);
|
|
}
|
|
|
|
static bool font_data_is_internal(font_data_t *fd)
|
|
{
|
|
return !REFCOUNT(fd); /* internal fonts have no reference counting */
|
|
}
|
|
|
|
static void font_data_free(font_data_t *fd)
|
|
{
|
|
kfree(to_font_data_struct(fd));
|
|
}
|
|
|
|
/**
|
|
* font_data_import - Allocates and initializes font data from user space
|
|
* @font: A font from user space
|
|
* @vpitch: The size of a single glyph in @font in bytes
|
|
* @calc_csum: An optional helper to calculate a chechsum
|
|
*
|
|
* Font data from user space must be translated to the kernel's format. The
|
|
* font's glyph geometry and data is provided in @font. The parameter @vpitch
|
|
* gives the number of bytes per glyph, including trailing bytes.
|
|
*
|
|
* The parameter @calc_csum is optional. Fbcon passes crc32() to calculate the
|
|
* font data's checksum.
|
|
*
|
|
* Returns:
|
|
* Newly initialized font data on success, or a pointer-encoded errno value otherwise.
|
|
*/
|
|
font_data_t *font_data_import(const struct console_font *font, unsigned int vpitch,
|
|
u32 (*calc_csum)(u32, const void *, size_t))
|
|
{
|
|
unsigned int pitch = console_font_pitch(font);
|
|
unsigned int h = font->height;
|
|
unsigned int charcount = font->charcount;
|
|
const unsigned char *data = font->data;
|
|
u32 csum = 0;
|
|
struct font_data *font_data;
|
|
int size, alloc_size;
|
|
unsigned int i;
|
|
font_data_t *fd;
|
|
|
|
/* Check for integer overflow in font-size calculation */
|
|
if (check_mul_overflow(h, pitch, &size) ||
|
|
check_mul_overflow(size, charcount, &size))
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
/* Check for overflow in allocation size calculation */
|
|
if (check_add_overflow(sizeof(*font_data), size, &alloc_size))
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
font_data = kmalloc(alloc_size, GFP_USER);
|
|
if (!font_data)
|
|
return ERR_PTR(-ENOMEM);
|
|
memset(font_data->extra, 0, sizeof(font_data->extra));
|
|
|
|
for (i = 0; i < charcount; ++i)
|
|
memcpy(font_data->data + i * h * pitch, data + i * vpitch * pitch, h * pitch);
|
|
|
|
if (calc_csum)
|
|
csum = calc_csum(0, font_data->data, size);
|
|
|
|
fd = font_data->data;
|
|
REFCOUNT(fd) = 1; /* start with reference acquired */
|
|
FNTSIZE(fd) = size;
|
|
FNTSUM(fd) = csum;
|
|
|
|
return fd;
|
|
}
|
|
EXPORT_SYMBOL_GPL(font_data_import);
|
|
|
|
/**
|
|
* font_data_get - Acquires a reference on font data
|
|
* @fd: Font data
|
|
*
|
|
* Font data from user space is reference counted. The helper
|
|
* font_data_get() increases the reference counter by one. Invoke
|
|
* font_data_put() to release the reference.
|
|
*
|
|
* Internal font data is located in read-only memory. In this case
|
|
* the helper returns success without modifying the counter field.
|
|
* It is still required to call font_data_put() on internal font data.
|
|
*/
|
|
void font_data_get(font_data_t *fd)
|
|
{
|
|
if (font_data_is_internal(fd))
|
|
return; /* never ref static data */
|
|
|
|
if (WARN_ON(!REFCOUNT(fd)))
|
|
return; /* should never be 0 */
|
|
++REFCOUNT(fd);
|
|
}
|
|
EXPORT_SYMBOL_GPL(font_data_get);
|
|
|
|
/**
|
|
* font_data_put - Release a reference on font data
|
|
* @fd: Font data
|
|
*
|
|
* Font data from user space is reference counted. The helper
|
|
* font_data_put() decreases the reference counter by one. If this was
|
|
* the final reference, it frees the allocated memory.
|
|
*
|
|
* Internal font data is located in read-only memory. In this case
|
|
* the helper returns success without modifying the counter field.
|
|
*
|
|
* Returns:
|
|
* True if the font data's memory buffer has been freed, false otherwise.
|
|
*/
|
|
bool font_data_put(font_data_t *fd)
|
|
{
|
|
unsigned int count;
|
|
|
|
if (font_data_is_internal(fd))
|
|
return false; /* never unref static data */
|
|
|
|
if (WARN_ON(!REFCOUNT(fd)))
|
|
return false; /* should never be 0 */
|
|
|
|
count = --REFCOUNT(fd);
|
|
if (!count)
|
|
font_data_free(fd);
|
|
|
|
return !count;
|
|
}
|
|
EXPORT_SYMBOL_GPL(font_data_put);
|
|
|
|
/**
|
|
* font_data_size - Return size of the font data in bytes
|
|
* @fd: Font data
|
|
*
|
|
* Returns:
|
|
* The number of bytes in the given font data.
|
|
*/
|
|
unsigned int font_data_size(font_data_t *fd)
|
|
{
|
|
return FNTSIZE(fd);
|
|
}
|
|
EXPORT_SYMBOL_GPL(font_data_size);
|
|
|
|
/**
|
|
* font_data_is_equal - Compares font data for equality
|
|
* @lhs: Left-hand side font data
|
|
* @rhs: Right-hand-size font data
|
|
*
|
|
* Font data is equal if is constain the same sequence of values. The
|
|
* helper also use the checksum, if both arguments contain it. Font data
|
|
* coming from different origins, internal or from user space, is never
|
|
* equal. Allowing this would break reference counting.
|
|
*
|
|
* Returns:
|
|
* True if the given font data is equal, false otherwise.
|
|
*/
|
|
bool font_data_is_equal(font_data_t *lhs, font_data_t *rhs)
|
|
{
|
|
if (font_data_is_internal(lhs) != font_data_is_internal(rhs))
|
|
return false;
|
|
if (font_data_size(lhs) != font_data_size(rhs))
|
|
return false;
|
|
if (FNTSUM(lhs) && FNTSUM(rhs) && FNTSUM(lhs) != FNTSUM(rhs))
|
|
return false;
|
|
|
|
return !memcmp(lhs, rhs, FNTSIZE(lhs));
|
|
}
|
|
EXPORT_SYMBOL_GPL(font_data_is_equal);
|
|
|
|
/**
|
|
* font_data_export - Stores font data for user space
|
|
* @fd: Font data
|
|
* @font: A font for user space
|
|
* @vpitch: The size of a single glyph in @font in bytes
|
|
*
|
|
* Store the font data given in @fd to the font in @font. Values and
|
|
* pointers in @font are pre-initialized. This helper mostly checks some
|
|
* corner cases and translates glyph sizes according to the value given
|
|
* @vpitch.
|
|
*
|
|
* Returns:
|
|
* 0 on success, or a negative errno code otherwise.
|
|
*/
|
|
int font_data_export(font_data_t *fd, struct console_font *font, unsigned int vpitch)
|
|
{
|
|
const unsigned char *font_data = font_data_buf(fd);
|
|
unsigned char *data = font->data;
|
|
unsigned int pitch = console_font_pitch(font);
|
|
unsigned int glyphsize, i;
|
|
|
|
if (!font->width || !font->height || !font->charcount || !font->data)
|
|
return 0;
|
|
|
|
glyphsize = font->height * pitch;
|
|
|
|
if (font->charcount * glyphsize > font_data_size(fd))
|
|
return -EINVAL;
|
|
|
|
for (i = 0; i < font->charcount; i++) {
|
|
memcpy(data, font_data, glyphsize);
|
|
memset(data + glyphsize, 0, pitch * vpitch - glyphsize);
|
|
data += pitch * vpitch;
|
|
font_data += glyphsize;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(font_data_export);
|
|
|
|
/*
|
|
* Font lookup
|
|
*/
|
|
|
|
static const struct font_desc *fonts[] = {
|
|
#ifdef CONFIG_FONT_8x8
|
|
&font_vga_8x8,
|
|
#endif
|
|
#ifdef CONFIG_FONT_8x16
|
|
&font_vga_8x16,
|
|
#endif
|
|
#ifdef CONFIG_FONT_6x11
|
|
&font_vga_6x11,
|
|
#endif
|
|
#ifdef CONFIG_FONT_7x14
|
|
&font_7x14,
|
|
#endif
|
|
#ifdef CONFIG_FONT_SUN8x16
|
|
&font_sun_8x16,
|
|
#endif
|
|
#ifdef CONFIG_FONT_SUN12x22
|
|
&font_sun_12x22,
|
|
#endif
|
|
#ifdef CONFIG_FONT_10x18
|
|
&font_10x18,
|
|
#endif
|
|
#ifdef CONFIG_FONT_ACORN_8x8
|
|
&font_acorn_8x8,
|
|
#endif
|
|
#ifdef CONFIG_FONT_PEARL_8x8
|
|
&font_pearl_8x8,
|
|
#endif
|
|
#ifdef CONFIG_FONT_MINI_4x6
|
|
&font_mini_4x6,
|
|
#endif
|
|
#ifdef CONFIG_FONT_6x10
|
|
&font_6x10,
|
|
#endif
|
|
#ifdef CONFIG_FONT_TER10x18
|
|
&font_ter_10x18,
|
|
#endif
|
|
#ifdef CONFIG_FONT_TER16x32
|
|
&font_ter_16x32,
|
|
#endif
|
|
#ifdef CONFIG_FONT_6x8
|
|
&font_6x8,
|
|
#endif
|
|
};
|
|
|
|
#define num_fonts ARRAY_SIZE(fonts)
|
|
|
|
#ifdef NO_FONTS
|
|
#error No fonts configured.
|
|
#endif
|
|
|
|
|
|
/**
|
|
* find_font - find a font
|
|
* @name: string name of a font
|
|
*
|
|
* Find a specified font with string name @name.
|
|
*
|
|
* Returns %NULL if no font found, or a pointer to the
|
|
* specified font.
|
|
*
|
|
*/
|
|
const struct font_desc *find_font(const char *name)
|
|
{
|
|
unsigned int i;
|
|
|
|
BUILD_BUG_ON(!num_fonts);
|
|
for (i = 0; i < num_fonts; i++)
|
|
if (!strcmp(fonts[i]->name, name))
|
|
return fonts[i];
|
|
return NULL;
|
|
}
|
|
EXPORT_SYMBOL(find_font);
|
|
|
|
|
|
/**
|
|
* get_default_font - get default font
|
|
* @xres: screen size of X
|
|
* @yres: screen size of Y
|
|
* @font_w: bit array of supported widths (1 - FB_MAX_BLIT_WIDTH)
|
|
* @font_h: bit array of supported heights (1 - FB_MAX_BLIT_HEIGHT)
|
|
*
|
|
* Get the default font for a specified screen size.
|
|
* Dimensions are in pixels.
|
|
*
|
|
* font_w or font_h being NULL means all values are supported.
|
|
*
|
|
* Returns %NULL if no font is found, or a pointer to the
|
|
* chosen font.
|
|
*
|
|
*/
|
|
const struct font_desc *get_default_font(int xres, int yres,
|
|
unsigned long *font_w,
|
|
unsigned long *font_h)
|
|
{
|
|
int i, c, cc, res;
|
|
const struct font_desc *f, *g;
|
|
|
|
g = NULL;
|
|
cc = -10000;
|
|
for (i = 0; i < num_fonts; i++) {
|
|
f = fonts[i];
|
|
c = f->pref;
|
|
#if defined(__mc68000__)
|
|
#ifdef CONFIG_FONT_PEARL_8x8
|
|
if (MACH_IS_AMIGA && f->idx == PEARL8x8_IDX)
|
|
c = 100;
|
|
#endif
|
|
#ifdef CONFIG_FONT_6x11
|
|
if (MACH_IS_MAC && xres < 640 && f->idx == VGA6x11_IDX)
|
|
c = 100;
|
|
#endif
|
|
#endif
|
|
if ((yres < 400) == (f->height <= 8))
|
|
c += 1000;
|
|
|
|
/* prefer a bigger font for high resolution */
|
|
res = (xres / f->width) * (yres / f->height) / 1000;
|
|
if (res > 20)
|
|
c += 20 - res;
|
|
|
|
if ((!font_w || test_bit(f->width - 1, font_w)) &&
|
|
(!font_h || test_bit(f->height - 1, font_h)))
|
|
c += 1000;
|
|
|
|
if (c > cc) {
|
|
cc = c;
|
|
g = f;
|
|
}
|
|
}
|
|
return g;
|
|
}
|
|
EXPORT_SYMBOL(get_default_font);
|
|
|
|
MODULE_AUTHOR("James Simmons <jsimmons@users.sf.net>");
|
|
MODULE_DESCRIPTION("Console Fonts");
|
|
MODULE_LICENSE("GPL");
|