mirror of
https://github.com/isometimes/rpi4-osdev
synced 2024-11-24 19:20:40 +00:00
Added loadspr.c with examples/wgt29.c which exercises it - a scrolly thing
This commit is contained in:
parent
6911e4630d
commit
c0f33240cb
7 changed files with 299 additions and 2 deletions
|
@ -17,11 +17,14 @@ bin/wgt1blk.o: bin/wgt1.blk
|
||||||
bin/wgt2blk.o: bin/wgt2.blk
|
bin/wgt2blk.o: bin/wgt2.blk
|
||||||
$(LLVMPATH)/llvm-objcopy -I binary -O elf64-littleaarch64 -B aarch64 $< $@
|
$(LLVMPATH)/llvm-objcopy -I binary -O elf64-littleaarch64 -B aarch64 $< $@
|
||||||
|
|
||||||
|
bin/lettersspr.o: bin/letters.spr
|
||||||
|
$(LLVMPATH)/llvm-objcopy -I binary -O elf64-littleaarch64 -B aarch64 $< $@
|
||||||
|
|
||||||
%.o: %.c
|
%.o: %.c
|
||||||
$(LLVMPATH)/clang --target=aarch64-elf $(CLANGFLAGS) -c $< -o $@
|
$(LLVMPATH)/clang --target=aarch64-elf $(CLANGFLAGS) -c $< -o $@
|
||||||
|
|
||||||
kernel8.img: boot/boot.o $(OFILES) bin/wgt1pal.o bin/wgt1blk.o bin/wgt2blk.o
|
kernel8.img: boot/boot.o $(OFILES) bin/wgt1pal.o bin/wgt1blk.o bin/wgt2blk.o bin/lettersspr.o
|
||||||
$(LLVMPATH)/ld.lld -m aarch64elf -nostdlib boot/boot.o $(OFILES) bin/wgt1pal.o bin/wgt1blk.o bin/wgt2blk.o -T boot/link.ld -o kernel8.elf
|
$(LLVMPATH)/ld.lld -m aarch64elf -nostdlib boot/boot.o $(OFILES) bin/wgt1pal.o bin/wgt1blk.o bin/wgt2blk.o bin/lettersspr.o -T boot/link.ld -o kernel8.elf
|
||||||
$(LLVMPATH)/llvm-objcopy -O binary kernel8.elf kernel8.img
|
$(LLVMPATH)/llvm-objcopy -O binary kernel8.elf kernel8.img
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
|
|
BIN
part12-wgt/bin/letters.spr
Normal file
BIN
part12-wgt/bin/letters.spr
Normal file
Binary file not shown.
110
part12-wgt/examples/wgt29.c
Normal file
110
part12-wgt/examples/wgt29.c
Normal file
|
@ -0,0 +1,110 @@
|
||||||
|
#include "wgt.h"
|
||||||
|
#include "include/mem.h"
|
||||||
|
|
||||||
|
// ######## REQUIRED FUNCTIONS ########
|
||||||
|
|
||||||
|
unsigned long state0 = 1000;
|
||||||
|
unsigned long state1 = 2000;
|
||||||
|
|
||||||
|
unsigned long rand(void)
|
||||||
|
{
|
||||||
|
unsigned long s1 = state0;
|
||||||
|
unsigned long s0 = state1;
|
||||||
|
|
||||||
|
state0 = s0;
|
||||||
|
s1 ^= s1 << 23;
|
||||||
|
s1 ^= s1 >> 17;
|
||||||
|
s1 ^= s0;
|
||||||
|
s1 ^= s0 >> 26;
|
||||||
|
state1 = s1;
|
||||||
|
|
||||||
|
return state0 + state1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void wait_msec(unsigned int n)
|
||||||
|
{
|
||||||
|
register unsigned long f, t, r;
|
||||||
|
|
||||||
|
// Get the current counter frequency
|
||||||
|
asm volatile ("mrs %0, cntfrq_el0" : "=r"(f));
|
||||||
|
// Read the current counter
|
||||||
|
asm volatile ("mrs %0, cntpct_el0" : "=r"(t));
|
||||||
|
// Calculate expire value for counter
|
||||||
|
t+=((f/1000)*n)/1000;
|
||||||
|
do{asm volatile ("mrs %0, cntpct_el0" : "=r"(r));}while(r<t);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ######## STUB FUNCTIONS ########
|
||||||
|
|
||||||
|
unsigned int kb = 0;
|
||||||
|
|
||||||
|
unsigned int kbhit(void) {
|
||||||
|
kb++;
|
||||||
|
return kb / 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
void getch(void) {
|
||||||
|
wait_msec(0x500000);
|
||||||
|
kb = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ######## WGT EXAMPLES ########
|
||||||
|
|
||||||
|
void wgt29()
|
||||||
|
{
|
||||||
|
color p[256]; /* Our palette */
|
||||||
|
block sprites[1001]; /* Pointers to blocks */
|
||||||
|
|
||||||
|
char *scrolltext="THIS IS A SMALL TEXT SCROLLER CREATED WITH THE WCOPYSCREEN COMMAND. PRESS A KEY TO END THIS SCROLLER... ";
|
||||||
|
|
||||||
|
int scrnum; /* Counter for current letter of scroller */
|
||||||
|
int nextlet, i, j; /* Counters */
|
||||||
|
|
||||||
|
set_clock_rate(get_max_clock());
|
||||||
|
mem_init();
|
||||||
|
vga256 (); /* Initialize graphics mode */
|
||||||
|
|
||||||
|
for (i = 0; i < 180; i++) /* Draws a background for the screen */
|
||||||
|
{
|
||||||
|
wsetcolor (vgapal[i]);
|
||||||
|
wline (0, 0, 319, i);
|
||||||
|
wline (319, 199, 0, 180 - i);
|
||||||
|
}
|
||||||
|
|
||||||
|
wsetcolor (vgapal[0]); /* Draw with black */
|
||||||
|
wbar (0, 189, 319, 199); /* Clear area off bottom for scroller */
|
||||||
|
|
||||||
|
extern unsigned char _binary_bin_letters_spr_start[];
|
||||||
|
unsigned char *lettersspr = &_binary_bin_letters_spr_start[0];
|
||||||
|
wloadsprites (p, lettersspr, sprites, 0, 1000); // Load our blocks
|
||||||
|
|
||||||
|
do {
|
||||||
|
scrnum = 0; /* Start at first character of scroller string */
|
||||||
|
do {
|
||||||
|
/* Find pixel width of current letter in string */
|
||||||
|
nextlet = wgetblockwidth (sprites[scrolltext[scrnum] + 1]);
|
||||||
|
for (j = 0; j <= nextlet + 1; j += 2)
|
||||||
|
{
|
||||||
|
wbar (318, 189, 319, 199); /* Erase right-hand side of scroller area */
|
||||||
|
|
||||||
|
/* Now copy the last letter on at the end of the string */
|
||||||
|
wputblock (319 - j, 189, sprites[scrolltext[scrnum] + 1], 0);
|
||||||
|
|
||||||
|
/* Wait for monitor to finish vertical retrace */
|
||||||
|
wait_msec(0x3E9F); // Ugly - this needs to sync better... could use the timer I suppose... NITV!
|
||||||
|
|
||||||
|
/* Shift scroller to the left */
|
||||||
|
wcopyscreen (2, 189, 319, 199, NULL, 0, 189, NULL);
|
||||||
|
}
|
||||||
|
scrnum++; /* Advance to next letter */
|
||||||
|
} while (scrolltext[scrnum + 1] != 0);
|
||||||
|
} while (1);
|
||||||
|
|
||||||
|
wfreesprites (sprites, 0, 1000); /* Free our sprites */
|
||||||
|
}
|
||||||
|
|
||||||
|
void main()
|
||||||
|
{
|
||||||
|
wgt29();
|
||||||
|
while (1);
|
||||||
|
}
|
90
part12-wgt/loadspr.c
Normal file
90
part12-wgt/loadspr.c
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
#include "wgt.h"
|
||||||
|
#include "include/mem.h"
|
||||||
|
|
||||||
|
short wloadsprites (color *pal, unsigned char *file, block *image_array, short start,
|
||||||
|
short end)
|
||||||
|
{
|
||||||
|
short maxsprite; /* maximum # of sprites in the file */
|
||||||
|
short startingsprite; /* First sprite in the file. Version <=3 = 1
|
||||||
|
Version 4 = 0 */
|
||||||
|
unsigned int a, b, i, spritemade;
|
||||||
|
char buf[14];
|
||||||
|
int size;
|
||||||
|
|
||||||
|
a = (unsigned int)*file;
|
||||||
|
file += 2;
|
||||||
|
|
||||||
|
/* Get the version number, and change the startingsprite accordingly.
|
||||||
|
If version <= 3, maxsprite contains the maximum number of sprites
|
||||||
|
that can be stored in a file. If version > 4, maxsprite contains
|
||||||
|
the number of the highest sprite in the file. (empty sprites at
|
||||||
|
the end are not kept in the file. */
|
||||||
|
if (a <= 3)
|
||||||
|
startingsprite = 1;
|
||||||
|
else
|
||||||
|
startingsprite = 0;
|
||||||
|
|
||||||
|
memcpy_char(&buf, file, 13);
|
||||||
|
file += 13;
|
||||||
|
|
||||||
|
if (0 == memcmp (" Sprite File ", &buf[0], 13))
|
||||||
|
/* see if it is a sprite file */
|
||||||
|
{
|
||||||
|
for (int k=0;k<256;k++) {
|
||||||
|
pal[k].r = 4 * *file++;
|
||||||
|
pal[k].g = 4 * *file++;
|
||||||
|
pal[k].b = 4 * *file++;
|
||||||
|
}
|
||||||
|
wsetpalette (0, 255, pal);
|
||||||
|
|
||||||
|
maxsprite = (short)*file; /* maximum sprites in this file */
|
||||||
|
file += 2;
|
||||||
|
|
||||||
|
for (i = start; i <= end; i++) /* make all sprites null
|
||||||
|
(still need to call wfreesprites) */
|
||||||
|
image_array[i] = NULL;
|
||||||
|
|
||||||
|
for (i = startingsprite; i <= maxsprite; i++) /* load them in */
|
||||||
|
{
|
||||||
|
spritemade = (unsigned int)*file; /* flag to see if sprite exists */
|
||||||
|
file += 2;
|
||||||
|
if (spritemade == 1)
|
||||||
|
{
|
||||||
|
a = (unsigned int)*file; /* get width and height */
|
||||||
|
file += 2;
|
||||||
|
b = (unsigned int)*file;
|
||||||
|
file += 2;
|
||||||
|
|
||||||
|
if ((i >= start) && (i <= end)) /* Load this one */
|
||||||
|
{
|
||||||
|
size = a * b;
|
||||||
|
|
||||||
|
image_array[i] = (block)malloc ((size * 4) + 12);
|
||||||
|
block arr = image_array[i];
|
||||||
|
|
||||||
|
*arr++ = a;
|
||||||
|
*arr++ = b;
|
||||||
|
|
||||||
|
for (int k=0;k<size;k++) *arr++ = vgapal[*file++];
|
||||||
|
}
|
||||||
|
else /* Don't load this one */
|
||||||
|
file += size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return (0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void wfreesprites (block *image_array, short start, short end)
|
||||||
|
{
|
||||||
|
short i;
|
||||||
|
|
||||||
|
for (i = start; i <= end; i++)
|
||||||
|
{
|
||||||
|
if (image_array[i] != NULL)
|
||||||
|
{
|
||||||
|
free (image_array[i]);
|
||||||
|
image_array[i] = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -45,6 +45,29 @@ void *memcpy_xray(void *dest, const void *src, unsigned len)
|
||||||
return dest;
|
return dest;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void *memcpy_char(void *dest, const void *src, unsigned len)
|
||||||
|
{
|
||||||
|
unsigned char *d = dest;
|
||||||
|
const unsigned char *s = src;
|
||||||
|
while (len--)
|
||||||
|
*d++ = *s++;
|
||||||
|
return dest;
|
||||||
|
}
|
||||||
|
|
||||||
|
int memcmp(char *str1, char *str2, unsigned count)
|
||||||
|
{
|
||||||
|
char *s1 = str1;
|
||||||
|
char *s2 = str2;
|
||||||
|
|
||||||
|
while (count-- > 0)
|
||||||
|
{
|
||||||
|
if (*s1++ != *s2++)
|
||||||
|
return s1[-1] < s2[-1] ? -1 : 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int abs(int i)
|
int abs(int i)
|
||||||
{
|
{
|
||||||
return i < 0 ? -i : i;
|
return i < 0 ? -i : i;
|
||||||
|
|
|
@ -56,6 +56,8 @@ extern short bx,by,tx,ty;
|
||||||
void *memset(void *dest, unsigned int val, unsigned len);
|
void *memset(void *dest, unsigned int val, unsigned len);
|
||||||
void *memcpy(void *dest, const void *src, unsigned len);
|
void *memcpy(void *dest, const void *src, unsigned len);
|
||||||
void *memcpy_xray(void *dest, const void *src, unsigned len);
|
void *memcpy_xray(void *dest, const void *src, unsigned len);
|
||||||
|
void *memcpy_char(void *dest, const void *src, unsigned len);
|
||||||
|
int memcmp(char *str1, char *str2, unsigned count);
|
||||||
int abs(int i);
|
int abs(int i);
|
||||||
int strlen(const char *str);
|
int strlen(const char *str);
|
||||||
|
|
||||||
|
@ -118,3 +120,6 @@ void wnormscreen (void);
|
||||||
void wvertres (short x, short y, short y2, block image);
|
void wvertres (short x, short y, short y2, block image);
|
||||||
void wwipe (short x, short y, short x2, short y2, block image);
|
void wwipe (short x, short y, short x2, short y2, block image);
|
||||||
void wskew (short x, short y, block image, short degrees);
|
void wskew (short x, short y, block image, short degrees);
|
||||||
|
short wloadsprites (color *pal, unsigned char *file, block *image_array, short start, short end);
|
||||||
|
void wfreesprites (block *image_array, short start, short end);
|
||||||
|
void wcopyscreen (short x, short y, short x2, short y2, block source, short destx, short desty, block dest);
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
#include "wgt.h"
|
#include "wgt.h"
|
||||||
|
|
||||||
|
#define fastcopy memcpy
|
||||||
|
|
||||||
void wsetscreen (block image)
|
void wsetscreen (block image)
|
||||||
{
|
{
|
||||||
if (image == NULL)
|
if (image == NULL)
|
||||||
|
@ -31,3 +33,67 @@ void wnormscreen (void)
|
||||||
bx = WGT_SYS.xres - 1;
|
bx = WGT_SYS.xres - 1;
|
||||||
by = WGT_SYS.yres - 1;
|
by = WGT_SYS.yres - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void wcopyscreen (short x, short y, short x2, short y2, block source, short destx, short desty, block dest)
|
||||||
|
{
|
||||||
|
unsigned int swidth, dwidth;
|
||||||
|
short width, height, display;
|
||||||
|
short ctr; /* Y line counter */
|
||||||
|
|
||||||
|
if (source == NULL)
|
||||||
|
{
|
||||||
|
source = abuf; /* Set to visual screen */
|
||||||
|
swidth = 320;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
swidth = *source;
|
||||||
|
source += 2; /* Skip over width and height */
|
||||||
|
}
|
||||||
|
|
||||||
|
width = abs (x - x2) + 1; /* Width of each row to copy */
|
||||||
|
height = abs (y - y2) + 1; /* Height of area to copy */
|
||||||
|
|
||||||
|
if (destx + width > bx)
|
||||||
|
display = bx + 1 - destx;
|
||||||
|
else display = width; /* Clip width */
|
||||||
|
|
||||||
|
if (height > (by + 1 - desty))
|
||||||
|
height = (by + 1 - desty);
|
||||||
|
|
||||||
|
source += x + (y * swidth); /* Clip height */
|
||||||
|
if (desty < ty)
|
||||||
|
{
|
||||||
|
source += (ty - desty) * swidth;
|
||||||
|
height -= ty - desty;
|
||||||
|
desty = ty;
|
||||||
|
} /* clip y */
|
||||||
|
|
||||||
|
if (destx < tx)
|
||||||
|
{
|
||||||
|
source += tx - destx;
|
||||||
|
display -= tx - destx;
|
||||||
|
destx = tx;
|
||||||
|
} /* clip x */
|
||||||
|
|
||||||
|
if (dest == NULL)
|
||||||
|
{
|
||||||
|
dest = abuf; /* Set to visual screen */
|
||||||
|
dwidth = 320;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
dwidth = *dest;
|
||||||
|
dest += 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Skip over width & height in image data */
|
||||||
|
dest += destx + (desty * dwidth); /* Find row offset in image data */
|
||||||
|
|
||||||
|
if ((height > 0) && (display > 0))
|
||||||
|
for (ctr = y; ctr < y + height; ctr++)
|
||||||
|
{
|
||||||
|
fastcopy (dest, source, display); /* Copy DISPLAY bytes to dest */
|
||||||
|
source += swidth; /* Advance one row */
|
||||||
|
dest += dwidth;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue