cregit-Linux how code gets into the kernel

Release 4.14 arch/powerpc/kernel/btext.c

// SPDX-License-Identifier: GPL-2.0
/*
 * Procedures for drawing on the screen early on in the boot process.
 *
 * Benjamin Herrenschmidt <benh@kernel.crashing.org>
 */
#include <linux/kernel.h>
#include <linux/string.h>
#include <linux/init.h>
#include <linux/export.h>
#include <linux/memblock.h>

#include <asm/sections.h>
#include <asm/prom.h>
#include <asm/btext.h>
#include <asm/page.h>
#include <asm/mmu.h>
#include <asm/pgtable.h>
#include <asm/io.h>
#include <asm/processor.h>
#include <asm/udbg.h>


#define NO_SCROLL

#ifndef NO_SCROLL
static void scrollscreen(void);
#endif


#define __force_data __attribute__((__section__(".data")))


static int g_loc_X __force_data;

static int g_loc_Y __force_data;

static int g_max_loc_X __force_data;

static int g_max_loc_Y __force_data;


static int dispDeviceRowBytes __force_data;

static int dispDeviceDepth  __force_data;

static int dispDeviceRect[4] __force_data;

static unsigned char *dispDeviceBase __force_data;

static unsigned char *logicalDisplayBase __force_data;


unsigned long disp_BAT[2] __initdata = {0, 0};


#define cmapsz	(16*256)


static unsigned char vga_font[cmapsz];


int boot_text_mapped __force_data = 0;

int force_printk_to_btext = 0;

extern void rmci_on(void);
extern void rmci_off(void);


static inline void rmci_maybe_on(void) { #if defined(CONFIG_PPC_EARLY_DEBUG_BOOTX) && defined(CONFIG_PPC64) if (!(mfmsr() & MSR_DR)) rmci_on(); #endif }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt35100.00%2100.00%
Total35100.00%2100.00%


static inline void rmci_maybe_off(void) { #if defined(CONFIG_PPC_EARLY_DEBUG_BOOTX) && defined(CONFIG_PPC64) if (!(mfmsr() & MSR_DR)) rmci_off(); #endif }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt35100.00%2100.00%
Total35100.00%2100.00%

#ifdef CONFIG_PPC32 /* Calc BAT values for mapping the display and store them * in disp_BAT. Those values are then used from head.S to map * the display during identify_machine() and MMU_Init() * * The display is mapped to virtual address 0xD0000000, rather * than 1:1, because some some CHRP machines put the frame buffer * in the region starting at 0xC0000000 (PAGE_OFFSET). * This mapping is temporary and will disappear as soon as the * setup done by MMU_Init() is applied. * * For now, we align the BAT and then map 8Mb on 601 and 16Mb * on other PPCs. This may cause trouble if the framebuffer * is really badly aligned, but I didn't encounter this case * yet. */
void __init btext_prepare_BAT(void) { unsigned long vaddr = PAGE_OFFSET + 0x10000000; unsigned long addr; unsigned long lowbits; addr = (unsigned long)dispDeviceBase; if (!addr) { boot_text_mapped = 0; return; } if (PVR_VER(mfspr(SPRN_PVR)) != 1) { /* 603, 604, G3, G4, ... */ lowbits = addr & ~0xFF000000UL; addr &= 0xFF000000UL; disp_BAT[0] = vaddr | (BL_16M<<2) | 2; disp_BAT[1] = addr | (_PAGE_NO_CACHE | _PAGE_GUARDED | BPP_RW); } else { /* 601 */ lowbits = addr & ~0xFF800000UL; addr &= 0xFF800000UL; disp_BAT[0] = vaddr | (_PAGE_NO_CACHE | PP_RWXX) | 4; disp_BAT[1] = addr | BL_8M | 0x40; } logicalDisplayBase = (void *) (vaddr + lowbits); }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras15299.35%150.00%
Michael Ellerman10.65%150.00%
Total153100.00%2100.00%

#endif /* This function can be used to enable the early boot text when doing * OF booting or within bootx init. It must be followed by a btext_unmap() * call before the logical address becomes unusable */
void __init btext_setup_display(int width, int height, int depth, int pitch, unsigned long address) { g_loc_X = 0; g_loc_Y = 0; g_max_loc_X = width / 8; g_max_loc_Y = height / 16; logicalDisplayBase = (unsigned char *)address; dispDeviceBase = (unsigned char *)address; dispDeviceRowBytes = pitch; dispDeviceDepth = depth == 15 ? 16 : depth; dispDeviceRect[0] = dispDeviceRect[1] = 0; dispDeviceRect[2] = width; dispDeviceRect[3] = height; boot_text_mapped = 1; }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras9894.23%150.00%
Benjamin Herrenschmidt65.77%150.00%
Total104100.00%2100.00%


void __init btext_unmap(void) { boot_text_mapped = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt12100.00%1100.00%
Total12100.00%1100.00%

/* Here's a small text engine to use during early boot * or for debugging purposes * * todo: * * - build some kind of vgacon with it to enable early printk * - move to a separate file * - add a few video driver hooks to keep in sync with display * changes. */
void btext_map(void) { unsigned long base, offset, size; unsigned char *vbase; /* By default, we are no longer mapped */ boot_text_mapped = 0; if (dispDeviceBase == 0) return; base = ((unsigned long) dispDeviceBase) & 0xFFFFF000UL; offset = ((unsigned long) dispDeviceBase) - base; size = dispDeviceRowBytes * dispDeviceRect[3] + offset + dispDeviceRect[0]; vbase = __ioremap(base, size, pgprot_val(pgprot_noncached_wc(__pgprot(0)))); if (vbase == 0) return; logicalDisplayBase = vbase + offset; boot_text_mapped = 1; }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras9889.91%133.33%
Aneesh Kumar K.V109.17%133.33%
Benjamin Herrenschmidt10.92%133.33%
Total109100.00%3100.00%


int btext_initialize(struct device_node *np) { unsigned int width, height, depth, pitch; unsigned long address = 0; const u32 *prop; prop = of_get_property(np, "linux,bootx-width", NULL); if (prop == NULL) prop = of_get_property(np, "width", NULL); if (prop == NULL) return -EINVAL; width = *prop; prop = of_get_property(np, "linux,bootx-height", NULL); if (prop == NULL) prop = of_get_property(np, "height", NULL); if (prop == NULL) return -EINVAL; height = *prop; prop = of_get_property(np, "linux,bootx-depth", NULL); if (prop == NULL) prop = of_get_property(np, "depth", NULL); if (prop == NULL) return -EINVAL; depth = *prop; pitch = width * ((depth + 7) / 8); prop = of_get_property(np, "linux,bootx-linebytes", NULL); if (prop == NULL) prop = of_get_property(np, "linebytes", NULL); if (prop && *prop != 0xffffffffu) pitch = *prop; if (pitch == 1) pitch = 0x1000; prop = of_get_property(np, "linux,bootx-addr", NULL); if (prop == NULL) prop = of_get_property(np, "address", NULL); if (prop) address = *prop; /* FIXME: Add support for PCI reg properties. Right now, only * reliable on macs */ if (address == 0) return -EINVAL; g_loc_X = 0; g_loc_Y = 0; g_max_loc_X = width / 8; g_max_loc_Y = height / 16; dispDeviceBase = (unsigned char *)address; dispDeviceRowBytes = pitch; dispDeviceDepth = depth == 15 ? 16 : depth; dispDeviceRect[0] = dispDeviceRect[1] = 0; dispDeviceRect[2] = width; dispDeviceRect[3] = height; btext_map(); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras24570.20%112.50%
Benjamin Herrenschmidt9426.93%562.50%
Stephen Rothwell92.58%112.50%
Jeremy Kerr10.29%112.50%
Total349100.00%8100.00%


int __init btext_find_display(int allow_nonstdout) { const char *name; struct device_node *np = NULL; int rc = -ENODEV; name = of_get_property(of_chosen, "linux,stdout-path", NULL); if (name != NULL) { np = of_find_node_by_path(name); if (np != NULL) { if (strcmp(np->type, "display") != 0) { printk("boot stdout isn't a display !\n"); of_node_put(np); np = NULL; } } } if (np) rc = btext_initialize(np); if (rc == 0 || !allow_nonstdout) return rc; for_each_node_by_type(np, "display") { if (of_get_property(np, "linux,opened", NULL)) { printk("trying %pOF ...\n", np); rc = btext_initialize(np); printk("result: %d\n", rc); } if (rc == 0) break; } return rc; }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras14587.88%116.67%
Benjamin Herrenschmidt148.48%116.67%
Cyrill V. Gorcunov21.21%116.67%
Stephen Rothwell21.21%116.67%
Jeremy Kerr10.61%116.67%
Rob Herring10.61%116.67%
Total165100.00%6100.00%

/* Calc the base address of a given point (x,y) */
static unsigned char * calc_base(int x, int y) { unsigned char *base; base = logicalDisplayBase; if (base == 0) base = dispDeviceBase; base += (x + dispDeviceRect[0]) * (dispDeviceDepth >> 3); base += (y + dispDeviceRect[1]) * dispDeviceRowBytes; return base; }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras66100.00%1100.00%
Total66100.00%1100.00%

/* Adjust the display to a new resolution */
void btext_update_display(unsigned long phys, int width, int height, int depth, int pitch) { if (dispDeviceBase == 0) return; /* check it's the same frame buffer (within 256MB) */ if ((phys ^ (unsigned long)dispDeviceBase) & 0xf0000000) return; dispDeviceBase = (__u8 *) phys; dispDeviceRect[0] = 0; dispDeviceRect[1] = 0; dispDeviceRect[2] = width; dispDeviceRect[3] = height; dispDeviceDepth = depth; dispDeviceRowBytes = pitch; if (boot_text_mapped) { iounmap(logicalDisplayBase); boot_text_mapped = 0; } btext_map(); g_loc_X = 0; g_loc_Y = 0; g_max_loc_X = width / 8; g_max_loc_Y = height / 16; }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras12599.21%150.00%
Benjamin Herrenschmidt10.79%150.00%
Total126100.00%2100.00%

EXPORT_SYMBOL(btext_update_display);
void btext_clearscreen(void) { unsigned int *base = (unsigned int *)calc_base(0, 0); unsigned long width = ((dispDeviceRect[2] - dispDeviceRect[0]) * (dispDeviceDepth >> 3)) >> 2; int i,j; rmci_maybe_on(); for (i=0; i<(dispDeviceRect[3] - dispDeviceRect[1]); i++) { unsigned int *ptr = base; for(j=width; j; --j) *(ptr++) = 0; base += (dispDeviceRowBytes >> 2); } rmci_maybe_off(); }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras10587.50%133.33%
Benjamin Herrenschmidt1512.50%266.67%
Total120100.00%3100.00%


void btext_flushscreen(void) { unsigned int *base = (unsigned int *)calc_base(0, 0); unsigned long width = ((dispDeviceRect[2] - dispDeviceRect[0]) * (dispDeviceDepth >> 3)) >> 2; int i,j; for (i=0; i < (dispDeviceRect[3] - dispDeviceRect[1]); i++) { unsigned int *ptr = base; for(j = width; j > 0; j -= 8) { __asm__ __volatile__ ("dcbst 0,%0" :: "r" (ptr)); ptr += 8; } base += (dispDeviceRowBytes >> 2); } __asm__ __volatile__ ("sync" ::: "memory"); }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt11796.69%150.00%
Paul Mackerras43.31%150.00%
Total121100.00%2100.00%


void btext_flushline(void) { unsigned int *base = (unsigned int *)calc_base(0, g_loc_Y << 4); unsigned long width = ((dispDeviceRect[2] - dispDeviceRect[0]) * (dispDeviceDepth >> 3)) >> 2; int i,j; for (i=0; i < 16; i++) { unsigned int *ptr = base; for(j = width; j > 0; j -= 8) { __asm__ __volatile__ ("dcbst 0,%0" :: "r" (ptr)); ptr += 8; } base += (dispDeviceRowBytes >> 2); } __asm__ __volatile__ ("sync" ::: "memory"); }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt11299.12%150.00%
Paul Mackerras10.88%150.00%
Total113100.00%2100.00%

#ifndef NO_SCROLL
static void scrollscreen(void) { unsigned int *src = (unsigned int *)calc_base(0,16); unsigned int *dst = (unsigned int *)calc_base(0,0); unsigned long width = ((dispDeviceRect[2] - dispDeviceRect[0]) * (dispDeviceDepth >> 3)) >> 2; int i,j; rmci_maybe_on(); for (i=0; i<(dispDeviceRect[3] - dispDeviceRect[1] - 16); i++) { unsigned int *src_ptr = src; unsigned int *dst_ptr = dst; for(j=width; j; --j) *(dst_ptr++) = *(src_ptr++); src += (dispDeviceRowBytes >> 2); dst += (dispDeviceRowBytes >> 2); } for (i=0; i<16; i++) { unsigned int *dst_ptr = dst; for(j=width; j; --j) *(dst_ptr++) = 0; dst += (dispDeviceRowBytes >> 2); } rmci_maybe_off(); }

Contributors

PersonTokensPropCommitsCommitProp
Paul Mackerras19191.83%133.33%
Benjamin Herrenschmidt178.17%266.67%
Total208100.00%3100.00%

#endif /* ndef NO_SCROLL */ static unsigned int expand_bits_8[16] = { 0x00000000, 0x000000ff, 0x0000ff00, 0x0000ffff, 0x00ff0000, 0x00ff00ff, 0x00ffff00, 0x00ffffff, 0xff000000, 0xff0000ff, 0xff00ff00, 0xff00ffff, 0xffff0000, 0xffff00ff, 0xffffff00, 0xffffffff }; static unsigned int expand_bits_16[4] = { 0x00000000, 0x0000ffff, 0xffff0000, 0xffffffff };
static void draw_byte_32(unsigned char *font, unsigned int *base, int rb) { int l, bits; int fg = 0xFFFFFFFFUL; int bg = 0x00000000UL; for (l = 0; l < 16; ++l) { bits = *font++; base[0] = (-(bits >> 7) & fg) ^ bg; base[1] = (-((bits >> 6) & 1) & fg) ^ bg; base[2] = (-((bits >> 5) & 1) & fg) ^ bg; base[3] = (-((bits >> 4) & 1) & fg) ^ bg; base[4] = (-((bits >> 3) & 1) & fg) ^ bg; base[5] = (-((bits >> 2) & 1) & fg) ^ bg; base[6] = (-((bits >> 1) & 1) & fg) ^ bg; base[7] = (-(bits & 1) & fg) ^ bg; base = (unsigned int *) ((char *)base + rb); } }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt20786.25%150.00%
Paul Mackerras3313.75%150.00%
Total240100.00%2100.00%


static inline void draw_byte_16(unsigned char *font, unsigned int *base, int rb) { int l, bits; int fg = 0xFFFFFFFFUL; int bg = 0x00000000UL; unsigned int *eb = (int *)expand_bits_16; for (l = 0; l < 16; ++l) { bits = *font++; base[0] = (eb[bits >> 6] & fg) ^ bg; base[1] = (eb[(bits >> 4) & 3] & fg) ^ bg; base[2] = (eb[(bits >> 2) & 3] & fg) ^ bg; base[3] = (eb[bits & 3] & fg) ^ bg; base = (unsigned int *) ((char *)base + rb); } }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt14990.85%150.00%
Paul Mackerras159.15%150.00%
Total164100.00%2100.00%


static inline void draw_byte_8(unsigned char *font, unsigned int *base, int rb) { int l, bits; int fg = 0x0F0F0F0FUL; int bg = 0x00000000UL; unsigned int *eb = (int *)expand_bits_8; for (l = 0; l < 16; ++l) { bits = *font++; base[0] = (eb[bits >> 4] & fg) ^ bg; base[1] = (eb[bits & 0xf] & fg) ^ bg; base = (unsigned int *) ((char *)base + rb); } }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt120100.00%2100.00%
Total120100.00%2100.00%


static noinline void draw_byte(unsigned char c, long locX, long locY) { unsigned char *base = calc_base(locX << 3, locY << 4); unsigned char *font = &vga_font[((unsigned int)c) * 16]; int rb = dispDeviceRowBytes; rmci_maybe_on(); switch(dispDeviceDepth) { case 24: case 32: draw_byte_32(font, (unsigned int *)base, rb); break; case 15: case 16: draw_byte_16(font, (unsigned int *)base, rb); break; case 8: draw_byte_8(font, (unsigned int *)base, rb); break; } rmci_maybe_off(); }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt10984.50%133.33%
Paul Mackerras1612.40%133.33%
Harvey Harrison43.10%133.33%
Total129100.00%3100.00%


void btext_drawchar(char c) { int cline = 0; #ifdef NO_SCROLL int x; #endif if (!boot_text_mapped) return; switch (c) { case '\b': if (g_loc_X > 0) --g_loc_X; break; case '\t': g_loc_X = (g_loc_X & -8) + 8; break; case '\r': g_loc_X = 0; break; case '\n': g_loc_X = 0; g_loc_Y++; cline = 1; break; default: draw_byte(c, g_loc_X++, g_loc_Y); } if (g_loc_X >= g_max_loc_X) { g_loc_X = 0; g_loc_Y++; cline = 1; } #ifndef NO_SCROLL while (g_loc_Y >= g_max_loc_Y) { scrollscreen(); g_loc_Y--; } #else /* wrap around from bottom to top of screen so we don't waste time scrolling each line. -- paulus. */ if (g_loc_Y >= g_max_loc_Y) g_loc_Y = 0; if (cline) { for (x = 0; x < g_max_loc_X; ++x) draw_byte(' ', x, g_loc_Y); } #endif }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt13879.31%150.00%
Paul Mackerras3620.69%150.00%
Total174100.00%2100.00%


void btext_drawstring(const char *c) { if (!boot_text_mapped) return; while (*c) btext_drawchar(*c++); }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt2692.86%150.00%
Paul Mackerras27.14%150.00%
Total28100.00%2100.00%


void btext_drawtext(const char *c, unsigned int len) { if (!boot_text_mapped) return; while (len--) btext_drawchar(*c++); }

Contributors

PersonTokensPropCommitsCommitProp
Benjamin Herrenschmidt2062.50%150.00%
Paul Mackerras1237.50%150.00%
Total32100.00%2100.00%


void btext_drawhex(unsigned long v) { if (!boot_text_mapped) return; #ifdef CONFIG_PPC64 btext_drawchar(hex_asc_hi(v >> 56)); btext_drawchar(hex_asc_lo(v >> 56)); btext_drawchar(hex_asc_hi(v >> 48)); btext_drawchar(hex_asc_lo(v >> 48)); btext_drawchar(hex_asc_hi(v >> 40)); btext_drawchar(hex_asc_lo(v >> 40))<