mirror of
https://github.com/DragonOS-Community/DragonOS.git
synced 2025-06-10 20:36:48 +00:00
🆕 完成了printk(暂不支持浮点数打印)
This commit is contained in:
parent
ebb5ef2141
commit
ead838bffd
4
.vscode/settings.json
vendored
4
.vscode/settings.json
vendored
@ -2,6 +2,8 @@
|
|||||||
"files.associations": {
|
"files.associations": {
|
||||||
"stdlib.h": "c",
|
"stdlib.h": "c",
|
||||||
"stdbool.h": "c",
|
"stdbool.h": "c",
|
||||||
"printk.h": "c"
|
"printk.h": "c",
|
||||||
|
"stdarg.h": "c",
|
||||||
|
"font.h": "c"
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,19 +1,27 @@
|
|||||||
|
SUBDIR_ROOTS := . common
|
||||||
|
DIRS := . $(shell find $(SUBDIR_ROOTS) -type d)
|
||||||
|
GARBAGE_PATTERNS := *.o *.s~ *.s *.S~ *.c~ *.h~ kernel *.a
|
||||||
|
GARBAGE := $(foreach DIR,$(DIRS),$(addprefix $(DIR)/,$(GARBAGE_PATTERNS)))
|
||||||
|
|
||||||
all: kernel
|
all: kernel
|
||||||
objcopy -I elf64-x86-64 -S -R ".eh_frame" -R ".comment" -O binary kernel ../bin/kernel/kernel.bin
|
objcopy -I elf64-x86-64 -S -R ".eh_frame" -R ".comment" -O binary kernel ../bin/kernel/kernel.bin
|
||||||
|
|
||||||
|
|
||||||
kernel: head.o main.o
|
kernel: head.o main.o printk.o
|
||||||
ld -b elf64-x86-64 -o kernel head.o main.o -T link.lds
|
ld -b elf64-x86-64 -z muldefs -o kernel head.o main.o printk.o -T link.lds
|
||||||
|
|
||||||
main.o: main.c
|
|
||||||
# -fno-builtin: 不使用C语言内建函数
|
|
||||||
# The -m64 option sets int to 32bits and long and pointer to 64 bits and generates code for AMD’s x86-64 architecture.
|
|
||||||
gcc -mcmodel=large -fno-builtin -m64 -c main.c
|
|
||||||
|
|
||||||
head.o: head.S
|
head.o: head.S
|
||||||
gcc -E head.S > head.s # 预处理
|
gcc -E head.S > head.s # 预处理
|
||||||
as --64 -o head.o head.s
|
as --64 -o head.o head.s
|
||||||
|
|
||||||
clean:
|
main.o: main.c
|
||||||
rm -rf *.o *.s~ *.s *.S~ *.c~ *.h~ kernel
|
# -fno-builtin: 不使用C语言内建函数
|
||||||
|
# The -m64 option sets int to 32bits and long and pointer to 64 bits and generates code for AMD’s x86-64 architecture.
|
||||||
|
gcc -mcmodel=large -fno-builtin -m64 -c main.c -fno-stack-protector
|
||||||
|
|
||||||
|
|
||||||
|
printk.o: common/printk.c
|
||||||
|
gcc -mcmodel=large -fno-builtin -m64 -c common/printk.c -fno-stack-protector
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -rf $(GARBAGE)
|
@ -4,9 +4,11 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
#ifndef GLIB_H
|
||||||
|
#define GLIB_H
|
||||||
|
|
||||||
//引入对bool类型的支持
|
//引入对bool类型的支持
|
||||||
#include<stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
#define NULL 0
|
#define NULL 0
|
||||||
|
|
||||||
@ -24,6 +26,8 @@
|
|||||||
#define io_lfence() __asm__ __volatile__("lfence\n\t" :: \
|
#define io_lfence() __asm__ __volatile__("lfence\n\t" :: \
|
||||||
: "memory") // 在lfence指令前的读操作当必须在lfence指令后的读操作前完成。
|
: "memory") // 在lfence指令前的读操作当必须在lfence指令后的读操作前完成。
|
||||||
|
|
||||||
|
#define ABS(x) ((x) > 0 ? (x) : -(x)) // 绝对值
|
||||||
|
|
||||||
//链表数据结构
|
//链表数据结构
|
||||||
struct List
|
struct List
|
||||||
{
|
{
|
||||||
@ -74,20 +78,21 @@ static inline void list_del(struct List *entry)
|
|||||||
entry->next = entry->prev;
|
entry->next = entry->prev;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline bool list_empty(struct List* entry)
|
static inline bool list_empty(struct List *entry)
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @brief 判断循环链表是否为空
|
* @brief 判断循环链表是否为空
|
||||||
* @param entry 入口
|
* @param entry 入口
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if(entry->prev == entry->next)
|
if (entry->prev == entry->next)
|
||||||
return true;
|
return true;
|
||||||
else return false;
|
else
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
//计算字符串的长度(经过测试,该版本比采用repne/scasb汇编的运行速度快16.8%左右)
|
//计算字符串的长度(经过测试,该版本比采用repne/scasb汇编的运行速度快16.8%左右)
|
||||||
static inline int strlen(char* s)
|
static inline int strlen(char *s)
|
||||||
{
|
{
|
||||||
register int __res = 0;
|
register int __res = 0;
|
||||||
while (s[__res] != '\0')
|
while (s[__res] != '\0')
|
||||||
@ -96,3 +101,5 @@ static inline int strlen(char* s)
|
|||||||
}
|
}
|
||||||
return __res;
|
return __res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#endif
|
@ -3,7 +3,66 @@
|
|||||||
//
|
//
|
||||||
#include "printk.h"
|
#include "printk.h"
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
//#include "linkage.h"
|
||||||
|
|
||||||
|
struct screen_info pos;
|
||||||
|
|
||||||
|
void show_color_band(int width, int height, char a, char b, char c, char d)
|
||||||
|
{
|
||||||
|
/** 向帧缓冲区写入像素值
|
||||||
|
* @param address: 帧缓存区的地址
|
||||||
|
* @param val:像素值
|
||||||
|
*/
|
||||||
|
|
||||||
|
for (int i = 0; i < width * height; ++i)
|
||||||
|
{
|
||||||
|
|
||||||
|
*((char *)pos.FB_address + 0) = d;
|
||||||
|
*((char *)pos.FB_address + 1) = c;
|
||||||
|
*((char *)pos.FB_address + 2) = b;
|
||||||
|
*((char *)pos.FB_address + 3) = a;
|
||||||
|
++pos.FB_address;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int calculate_max_charNum(int len, int size)
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @brief 计算屏幕上能有多少行
|
||||||
|
* @param len 屏幕长/宽
|
||||||
|
* @param size 字符长/宽
|
||||||
|
*/
|
||||||
|
return len / size;
|
||||||
|
}
|
||||||
|
|
||||||
|
int init_printk(const int width, const int height, unsigned int *FB_address, const int FB_length, const int char_size_x, const int char_size_y)
|
||||||
|
{
|
||||||
|
|
||||||
|
pos.width = width;
|
||||||
|
pos.height = height;
|
||||||
|
pos.char_size_x = char_size_x;
|
||||||
|
pos.char_size_y = char_size_y;
|
||||||
|
pos.max_x = calculate_max_charNum(width, char_size_x);
|
||||||
|
pos.max_y = calculate_max_charNum(height, char_size_y);
|
||||||
|
|
||||||
|
pos.FB_address = FB_address;
|
||||||
|
pos.FB_length = FB_length;
|
||||||
|
|
||||||
|
pos.x = 0;
|
||||||
|
pos.y = 0;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int set_printk_pos(const int x, const int y)
|
||||||
|
{
|
||||||
|
// 指定的坐标不在屏幕范围内
|
||||||
|
if (!((x >= 0 && x <= pos.max_x) && (y >= 0 && y <= pos.max_y)))
|
||||||
|
return EPOS_OVERFLOW;
|
||||||
|
pos.x = x;
|
||||||
|
pos.y = y;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
int skip_and_atoi(const char **s)
|
int skip_and_atoi(const char **s)
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
@ -18,6 +77,22 @@ int skip_and_atoi(const char **s)
|
|||||||
}
|
}
|
||||||
return ans;
|
return ans;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void auto_newline()
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @brief 超过每行最大字符数,自动换行
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
if (pos.x > pos.max_x)
|
||||||
|
{
|
||||||
|
pos.x = 0;
|
||||||
|
++pos.y;
|
||||||
|
}
|
||||||
|
if (pos.y > pos.max_y)
|
||||||
|
pos.y = 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int vsprintf(char *buf, const char *fmt, va_list args)
|
static int vsprintf(char *buf, const char *fmt, va_list args)
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
@ -53,11 +128,11 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
|
|||||||
|
|
||||||
//清空标志位和field宽度
|
//清空标志位和field宽度
|
||||||
field_width = flags = 0;
|
field_width = flags = 0;
|
||||||
++fmt;
|
|
||||||
|
|
||||||
bool flag_tmp = true;
|
bool flag_tmp = true;
|
||||||
bool flag_break = false;
|
bool flag_break = false;
|
||||||
|
|
||||||
|
++fmt;
|
||||||
while (flag_tmp)
|
while (flag_tmp)
|
||||||
{
|
{
|
||||||
switch (*fmt)
|
switch (*fmt)
|
||||||
@ -67,13 +142,7 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
|
|||||||
flag_break = true;
|
flag_break = true;
|
||||||
flag_tmp = false;
|
flag_tmp = false;
|
||||||
break;
|
break;
|
||||||
case '%':
|
|
||||||
//输出 %
|
|
||||||
*str = '%';
|
|
||||||
++str;
|
|
||||||
++fmt;
|
|
||||||
flag_break = true;
|
|
||||||
break;
|
|
||||||
case '-':
|
case '-':
|
||||||
// 左对齐
|
// 左对齐
|
||||||
flags |= LEFT;
|
flags |= LEFT;
|
||||||
@ -107,13 +176,21 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
//获取区域宽度
|
//获取区域宽度
|
||||||
|
field_width = -1;
|
||||||
if (*fmt == '*')
|
if (*fmt == '*')
|
||||||
{
|
{
|
||||||
field_width = va_arg(args, int);
|
field_width = va_arg(args, int);
|
||||||
++fmt;
|
++fmt;
|
||||||
}
|
}
|
||||||
else if (is_digit(*fmt))
|
else if (is_digit(*fmt))
|
||||||
|
{
|
||||||
field_width = skip_and_atoi(&fmt);
|
field_width = skip_and_atoi(&fmt);
|
||||||
|
if (field_width < 0)
|
||||||
|
{
|
||||||
|
field_width = -field_width;
|
||||||
|
flags |= LEFT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//获取小数精度
|
//获取小数精度
|
||||||
precision = -1;
|
precision = -1;
|
||||||
@ -137,29 +214,34 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
|
|||||||
qualifier = *fmt;
|
qualifier = *fmt;
|
||||||
++fmt;
|
++fmt;
|
||||||
}
|
}
|
||||||
|
//为了支持lld
|
||||||
|
if (qualifier == 'l' && *fmt == 'l', *(fmt + 1) == 'd')
|
||||||
|
++fmt;
|
||||||
|
|
||||||
//转化成字符串
|
//转化成字符串
|
||||||
|
long long *ip;
|
||||||
switch (*fmt)
|
switch (*fmt)
|
||||||
{
|
{
|
||||||
|
//输出 %
|
||||||
|
case '%':
|
||||||
|
*str++ = '%';
|
||||||
|
|
||||||
|
break;
|
||||||
// 显示一个字符
|
// 显示一个字符
|
||||||
case 'c':
|
case 'c':
|
||||||
//靠右对齐
|
//靠右对齐
|
||||||
if (!(flags & LEFT))
|
if (!(flags & LEFT))
|
||||||
{
|
{
|
||||||
while (--field_width)
|
while (--field_width > 0)
|
||||||
{
|
{
|
||||||
*str = ' ';
|
*str = ' ';
|
||||||
++str;
|
++str;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else //靠左对齐
|
|
||||||
{
|
*str++ = (unsigned char)va_arg(args, int);
|
||||||
*str = (char)va_arg(args, int);
|
|
||||||
++str;
|
while (--field_width > 0)
|
||||||
--field_width;
|
|
||||||
}
|
|
||||||
while (--field_width)
|
|
||||||
{
|
{
|
||||||
*str = ' ';
|
*str = ' ';
|
||||||
++str;
|
++str;
|
||||||
@ -208,10 +290,13 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
|
|||||||
break;
|
break;
|
||||||
//以八进制显示字符串
|
//以八进制显示字符串
|
||||||
case 'o':
|
case 'o':
|
||||||
|
flags |= SMALL;
|
||||||
|
case 'O':
|
||||||
|
flags |= SPECIAL;
|
||||||
if (qualifier == 'l')
|
if (qualifier == 'l')
|
||||||
write_num(str, va_arg(args, long long), 8, field_width, precision, flags);
|
str = write_num(str, va_arg(args, long long), 8, field_width, precision, flags);
|
||||||
else
|
else
|
||||||
write_num(str, va_arg(args, int), 8, field_width, precision, flags);
|
str = write_num(str, va_arg(args, int), 8, field_width, precision, flags);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
//打印指针指向的地址
|
//打印指针指向的地址
|
||||||
@ -222,7 +307,7 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
|
|||||||
flags |= PAD_ZERO;
|
flags |= PAD_ZERO;
|
||||||
}
|
}
|
||||||
|
|
||||||
write_num(str, (unsigned long)va_arg(args, void *), 16, field_width, precision, flags);
|
str = write_num(str, (unsigned long)va_arg(args, void *), 16, field_width, precision, flags);
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -230,34 +315,35 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
|
|||||||
case 'x':
|
case 'x':
|
||||||
flags |= SMALL;
|
flags |= SMALL;
|
||||||
case 'X':
|
case 'X':
|
||||||
|
flags |= SPECIAL;
|
||||||
if (qualifier == 'l')
|
if (qualifier == 'l')
|
||||||
write_num(str, va_arg(args, long long), 16, field_width, precision, flags);
|
str = write_num(str, va_arg(args, long long), 16, field_width, precision, flags);
|
||||||
else
|
else
|
||||||
write_num(str, va_arg(args, int), 16, field_width, precision, flags);
|
str = write_num(str, va_arg(args, int), 16, field_width, precision, flags);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
//打印十进制有符号整数
|
//打印十进制有符号整数
|
||||||
case 'i':
|
case 'i':
|
||||||
case 'd':
|
case 'd':
|
||||||
case 'ld':
|
|
||||||
flags |= SIGN;
|
flags |= SIGN;
|
||||||
if (qualifier == 'l')
|
if (qualifier == 'l')
|
||||||
write_num(str, va_arg(args, long long), 10, field_width, precision, flags);
|
str = write_num(str, va_arg(args, long long), 10, field_width, precision, flags);
|
||||||
else
|
else
|
||||||
write_num(str, va_arg(args, int), 10, field_width, precision, flags);
|
str = write_num(str, va_arg(args, int), 10, field_width, precision, flags);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
//打印十进制无符号整数
|
//打印十进制无符号整数
|
||||||
case 'u':
|
case 'u':
|
||||||
if (qualifier == 'l')
|
if (qualifier == 'l')
|
||||||
write_num(str, va_arg(args, unsigned long long), 10, field_width, precision, flags);
|
str = write_num(str, va_arg(args, unsigned long long), 10, field_width, precision, flags);
|
||||||
else
|
else
|
||||||
write_num(str, va_arg(args, unsigned int), 10, field_width, precision, flags);
|
str = write_num(str, va_arg(args, unsigned int), 10, field_width, precision, flags);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
//输出有效字符数量到*ip对应的变量
|
//输出有效字符数量到*ip对应的变量
|
||||||
case 'n':
|
case 'n':
|
||||||
long long *ip;
|
|
||||||
if (qualifier == 'l')
|
if (qualifier == 'l')
|
||||||
ip = va_arg(args, long long *);
|
ip = va_arg(args, long long *);
|
||||||
else
|
else
|
||||||
@ -269,18 +355,20 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
|
|||||||
//对于不识别的控制符,直接输出
|
//对于不识别的控制符,直接输出
|
||||||
default:
|
default:
|
||||||
*str++ = '%';
|
*str++ = '%';
|
||||||
if(*fmt)
|
if (*fmt)
|
||||||
*str++ = *fmt;
|
*str++ = *fmt;
|
||||||
else --fmt;
|
else
|
||||||
|
--fmt;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
*str = '\0';
|
*str = '\0';
|
||||||
|
|
||||||
//返回缓冲区已有字符串的长度。
|
//返回缓冲区已有字符串的长度。
|
||||||
return str-buf;
|
return str - buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void write_num(char *str, long long num, int base, int field_width, int precision, int flags)
|
static char *write_num(char *str, long long num, int base, int field_width, int precision, int flags)
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @brief 将数字按照指定的要求转换成对应的字符串
|
* @brief 将数字按照指定的要求转换成对应的字符串
|
||||||
@ -296,7 +384,6 @@ static void write_num(char *str, long long num, int base, int field_width, int p
|
|||||||
// 首先判断是否支持该进制
|
// 首先判断是否支持该进制
|
||||||
if (base < 2 || base > 36)
|
if (base < 2 || base > 36)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
char pad, sign, tmp_num[100];
|
char pad, sign, tmp_num[100];
|
||||||
|
|
||||||
const char *digits = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
const char *digits = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
||||||
@ -335,11 +422,11 @@ static void write_num(char *str, long long num, int base, int field_width, int p
|
|||||||
tmp_num[js_num++] = '0';
|
tmp_num[js_num++] = '0';
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
num = abs(num);
|
num = ABS(num);
|
||||||
//进制转换
|
//进制转换
|
||||||
while (num)
|
while (num > 0)
|
||||||
{
|
{
|
||||||
tmp_num[js_num++] = num % base; // 注意这里,输出的数字,是小端对齐的。低位存低位
|
tmp_num[js_num++] = digits[num % base]; // 注意这里,输出的数字,是小端对齐的。低位存低位
|
||||||
num /= base;
|
num /= base;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -351,7 +438,7 @@ static void write_num(char *str, long long num, int base, int field_width, int p
|
|||||||
|
|
||||||
// 靠右对齐
|
// 靠右对齐
|
||||||
if (!(flags & LEFT))
|
if (!(flags & LEFT))
|
||||||
while (field_width--)
|
while (field_width-- > 0)
|
||||||
*str++ = pad;
|
*str++ = pad;
|
||||||
|
|
||||||
if (sign)
|
if (sign)
|
||||||
@ -371,11 +458,119 @@ static void write_num(char *str, long long num, int base, int field_width, int p
|
|||||||
*str++ = '0';
|
*str++ = '0';
|
||||||
}
|
}
|
||||||
|
|
||||||
while (js_num--)
|
while (js_num-- > 0)
|
||||||
*str++ = tmp_num[js_num];
|
*str++ = tmp_num[js_num];
|
||||||
|
|
||||||
|
|
||||||
while (field_width--)
|
while (field_width-- > 0)
|
||||||
*str++ = ' ';
|
*str++ = ' ';
|
||||||
|
|
||||||
return str;
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void putchar(unsigned int *fb, int Xsize, int x, int y, unsigned int FRcolor, unsigned int BKcolor, unsigned char font)
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @brief 在屏幕上指定位置打印字符
|
||||||
|
*
|
||||||
|
* @param fb 帧缓存线性地址
|
||||||
|
* @param Xsize 行分辨率
|
||||||
|
* @param x 左上角列像素点位置
|
||||||
|
* @param y 左上角行像素点位置
|
||||||
|
* @param FRcolor 字体颜色
|
||||||
|
* @param BKcolor 背景颜色
|
||||||
|
* @param font 字符的bitmap
|
||||||
|
*/
|
||||||
|
|
||||||
|
unsigned char *font_ptr = font_ascii[font];
|
||||||
|
unsigned int *addr;
|
||||||
|
|
||||||
|
int testbit; // 用来测试某位是背景还是字体本身
|
||||||
|
|
||||||
|
for (int i = 0; i < pos.char_size_y; ++i)
|
||||||
|
{
|
||||||
|
// 计算出帧缓冲区的地址
|
||||||
|
addr = fb + Xsize * (y + i) + x;
|
||||||
|
testbit = (1 << (pos.char_size_x + 1));
|
||||||
|
for (int j = 0; j < pos.char_size_x; ++j)
|
||||||
|
{
|
||||||
|
//从左往右逐个测试相应位
|
||||||
|
testbit >>= 1;
|
||||||
|
if (*font_ptr & testbit)
|
||||||
|
*addr = FRcolor; // 字,显示前景色
|
||||||
|
else
|
||||||
|
*addr = BKcolor; // 背景色
|
||||||
|
|
||||||
|
++addr;
|
||||||
|
}
|
||||||
|
++font_ptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int printk_color(unsigned int FRcolor, unsigned int BKcolor, const char *fmt, ...)
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @brief 格式化打印字符串
|
||||||
|
*
|
||||||
|
* @param FRcolor 前景色
|
||||||
|
* @param BKcolor 背景色
|
||||||
|
* @param ... 格式化字符串
|
||||||
|
*/
|
||||||
|
|
||||||
|
va_list args;
|
||||||
|
va_start(args, fmt);
|
||||||
|
|
||||||
|
int len = vsprintf(buf, fmt, args);
|
||||||
|
|
||||||
|
va_end(args);
|
||||||
|
unsigned char current;
|
||||||
|
|
||||||
|
int i; // 总共输出的字符数
|
||||||
|
for (i = 0; i < len; ++i)
|
||||||
|
{
|
||||||
|
current = *(buf + i);
|
||||||
|
//输出换行
|
||||||
|
if (current == '\n')
|
||||||
|
{
|
||||||
|
pos.x = 0;
|
||||||
|
++pos.y;
|
||||||
|
}
|
||||||
|
else if (current == '\t') // 输出制表符
|
||||||
|
{
|
||||||
|
int space_to_print = 8 - pos.x % 8;
|
||||||
|
|
||||||
|
while (space_to_print--)
|
||||||
|
{
|
||||||
|
putchar(pos.FB_address, pos.width, pos.x * pos.char_size_x, pos.y * pos.char_size_y, BLACK, BLACK, ' ');
|
||||||
|
++pos.x;
|
||||||
|
|
||||||
|
auto_newline();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (current == '\b') // 退格
|
||||||
|
{
|
||||||
|
--pos.x;
|
||||||
|
if (pos.x < 0)
|
||||||
|
{
|
||||||
|
--pos.y;
|
||||||
|
if (pos.y <= 0)
|
||||||
|
pos.x = pos.y = 0;
|
||||||
|
else
|
||||||
|
pos.x = pos.max_x;
|
||||||
|
}
|
||||||
|
|
||||||
|
putchar(pos.FB_address, pos.width, pos.x * pos.char_size_x, pos.y * pos.char_size_y, FRcolor, BKcolor, ' ');
|
||||||
|
++pos.x;
|
||||||
|
|
||||||
|
auto_newline();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
putchar(pos.FB_address, pos.width, pos.x * pos.char_size_x, pos.y * pos.char_size_y, FRcolor, BKcolor, current);
|
||||||
|
++pos.x;
|
||||||
|
auto_newline();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
}
|
}
|
124
kernel/common/printk.h
Normal file
124
kernel/common/printk.h
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
//
|
||||||
|
// Created by longjin on 2022/1/21.
|
||||||
|
//
|
||||||
|
#pragma once
|
||||||
|
#ifndef PRINTK_H
|
||||||
|
#define PRINTK_H
|
||||||
|
|
||||||
|
#define PAD_ZERO 1 // 0填充
|
||||||
|
#define LEFT 2 // 靠左对齐
|
||||||
|
#define RIGHT 4 // 靠右对齐
|
||||||
|
#define PLUS 8 // 在正数前面显示加号
|
||||||
|
#define SPACE 16
|
||||||
|
#define SPECIAL 32 // 在八进制数前面显示 '0o',在十六进制数前面显示 '0x' 或 '0X'
|
||||||
|
#define SMALL 64 // 十进制以上数字显示小写字母
|
||||||
|
#define SIGN 128 // 显示符号位
|
||||||
|
|
||||||
|
#define is_digit(c) ((c) >= '0' && (c) <= '9') // 用来判断是否是数字的宏
|
||||||
|
|
||||||
|
// 字体颜色的宏定义
|
||||||
|
#define WHITE 0x00ffffff //白
|
||||||
|
#define BLACK 0x00000000 //黑
|
||||||
|
#define RED 0x00ff0000 //红
|
||||||
|
#define ORANGE 0x00ff8000 //橙
|
||||||
|
#define YELLOW 0x00ffff00 //黄
|
||||||
|
#define GREEN 0x0000ff00 //绿
|
||||||
|
#define BLUE 0x000000ff //蓝
|
||||||
|
#define INDIGO 0x0000ffff //靛
|
||||||
|
#define PURPLE 0x008000ff //紫
|
||||||
|
|
||||||
|
// 异常的宏定义
|
||||||
|
#define EPOS_OVERFLOW 1 // 坐标溢出
|
||||||
|
#define EFB_MISMATCH 2 // 帧缓冲区与指定的屏幕大小不匹配
|
||||||
|
|
||||||
|
#include "font.h"
|
||||||
|
#include "glib.h"
|
||||||
|
//#include "linkage.h"
|
||||||
|
#include <stdarg.h>
|
||||||
|
|
||||||
|
struct screen_info
|
||||||
|
{
|
||||||
|
int width, height; //屏幕大小
|
||||||
|
|
||||||
|
int max_x, max_y; // 最大x、y字符数
|
||||||
|
|
||||||
|
int x, y; //光标位置
|
||||||
|
|
||||||
|
int char_size_x, char_size_y;
|
||||||
|
|
||||||
|
unsigned int *FB_address; //帧缓冲区首地址
|
||||||
|
unsigned long FB_length; // 帧缓冲区长度
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
extern unsigned char font_ascii[256][16]; //导出ascii字体的bitmap(8*16大小) ps:位于font.h中
|
||||||
|
|
||||||
|
char buf[4096]; //vsprintf()的缓冲区
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief 初始化printk的屏幕信息
|
||||||
|
*
|
||||||
|
* @param width 屏幕宽度
|
||||||
|
* @param height 屏幕高度
|
||||||
|
* @param FB_address 帧缓冲区地址
|
||||||
|
* @param FB_length 帧缓冲区长度
|
||||||
|
* @param char_size_x 字符的列坐标
|
||||||
|
* @param char_size_y 字符的行坐标
|
||||||
|
*/
|
||||||
|
int init_printk(const int width, const int height, unsigned int *FB_address, const int FB_length, const int char_size_x, const int char_size_y);
|
||||||
|
/**
|
||||||
|
* @brief Set the printk pos object
|
||||||
|
*
|
||||||
|
* @param x 列坐标
|
||||||
|
* @param y 行坐标
|
||||||
|
*/
|
||||||
|
static int set_printk_pos(const int x, const int y);
|
||||||
|
/**
|
||||||
|
* @brief 将字符串按照fmt和args中的内容进行格式化,然后保存到buf中
|
||||||
|
*
|
||||||
|
* @param buf 结果缓冲区
|
||||||
|
* @param fmt 格式化字符串
|
||||||
|
* @param args 内容
|
||||||
|
* @return 最终字符串的长度
|
||||||
|
*/
|
||||||
|
static int vsprintf(char *buf, const char *fmt, va_list args);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief 将数字按照指定的要求转换成对应的字符串(2~36进制)
|
||||||
|
*
|
||||||
|
* @param str 要返回的字符串
|
||||||
|
* @param num 要打印的数值
|
||||||
|
* @param base 基数
|
||||||
|
* @param field_width 区域宽度
|
||||||
|
* @param precision 精度
|
||||||
|
* @param flags 标志位
|
||||||
|
*/
|
||||||
|
static char* write_num(char *str, long long num, int base, int field_width, int precision, int flags);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief 在屏幕上指定位置打印字符
|
||||||
|
*
|
||||||
|
* @param fb 帧缓存线性地址
|
||||||
|
* @param Xsize 行分辨率
|
||||||
|
* @param x 左上角列像素点位置
|
||||||
|
* @param y 左上角行像素点位置
|
||||||
|
* @param FRcolor 字体颜色
|
||||||
|
* @param BKcolor 背景颜色
|
||||||
|
* @param font 字符的bitmap
|
||||||
|
*/
|
||||||
|
static void putchar(unsigned int *fb, int Xsize, int x, int y, unsigned int FRcolor, unsigned int BKcolor, unsigned char font);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief 格式化打印字符串
|
||||||
|
*
|
||||||
|
* @param FRcolor 前景色
|
||||||
|
* @param BKcolor 背景色
|
||||||
|
* @param ... 格式化字符串
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#define printk(...) printk_color( WHITE, BLACK, __VA_ARGS__ )
|
||||||
|
|
||||||
|
int printk_color(unsigned int FRcolor, unsigned int BKcolor, const char*fmt, ...);
|
||||||
|
|
||||||
|
#endif
|
@ -1,41 +1,70 @@
|
|||||||
//
|
//
|
||||||
// Created by longjin on 2022/1/20.
|
// Created by longjin on 2022/1/20.
|
||||||
//
|
//
|
||||||
int *address = (int *)0xffff800000a00000; //帧缓存区的地址
|
|
||||||
|
|
||||||
void show_color_band(int width, int height, char a, char b, char c, char d)
|
#include "common/glib.h"
|
||||||
|
#include "common/printk.h"
|
||||||
|
|
||||||
|
int *FR_address = (int *)0xffff800000a00000; //帧缓存区的地址
|
||||||
|
|
||||||
|
void show_welcome()
|
||||||
{
|
{
|
||||||
/** 向帧缓冲区写入像素值
|
/**
|
||||||
* @param address: 帧缓存区的地址
|
* @brief 打印欢迎页面
|
||||||
* @param val:像素值
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
for (int i = 0; i < width * height; ++i)
|
printk("\n\n");
|
||||||
{
|
for (int i = 0; i < 74; ++i)
|
||||||
|
printk(" ");
|
||||||
*((char *)address + 0) = d;
|
printk_color(0x00e0ebeb, 0x00e0ebeb, " \n");
|
||||||
*((char *)address + 1) = c;
|
for (int i = 0; i < 74; ++i)
|
||||||
*((char *)address + 2) = b;
|
printk(" ");
|
||||||
*((char *)address + 3) = a;
|
printk_color(BLACK, 0x00e0ebeb, " Welcome to DragonOS ! \n");
|
||||||
++address;
|
for (int i = 0; i < 74; ++i)
|
||||||
}
|
printk(" ");
|
||||||
|
printk_color(0x00e0ebeb, 0x00e0ebeb, " \n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void test_printk()
|
||||||
|
{
|
||||||
|
//测试直接输出
|
||||||
|
printk("\nTesting printk...\n");
|
||||||
|
//测试输出单个字符
|
||||||
|
printk("%c\n", 't');
|
||||||
|
//测试输出字符串%s
|
||||||
|
printk("%s\n", "xxx");
|
||||||
|
|
||||||
|
//测试输出数字
|
||||||
|
printk("%d %ld %lld\n", 1, 2, 3);
|
||||||
|
|
||||||
|
//测试输出两个百分号
|
||||||
|
printk("%%\n");
|
||||||
|
|
||||||
|
//测试输出\t
|
||||||
|
printk("\nTesting tab...\n");
|
||||||
|
printk("date\t\tname\tscore\n");
|
||||||
|
printk("2022-01-01\tDavid\t99\n");
|
||||||
|
printk("2022-01-01\tJohn\t95\n");
|
||||||
|
|
||||||
|
|
||||||
|
//测试输出八进制
|
||||||
|
printk("\nTest base 8 : %d --> %o\n", 255, 255);
|
||||||
|
|
||||||
|
//测试输出十六进制
|
||||||
|
printk("\nTest base 16 : %d --> %x\n", 255, 255);
|
||||||
|
printk("\nTest base 16 : %d --> %X\n", 255, 255);
|
||||||
|
}
|
||||||
//操作系统内核从这里开始执行
|
//操作系统内核从这里开始执行
|
||||||
void Start_Kernel(void)
|
void Start_Kernel(void)
|
||||||
{
|
{
|
||||||
|
// 初始化printk
|
||||||
|
init_printk(1440, 900, FR_address, 1440 * 900 * 4, 8, 16);
|
||||||
|
|
||||||
|
show_welcome();
|
||||||
|
test_printk();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
show_color_band(1440, 20, 0x00, 0xff, 0x00, 0x00);
|
|
||||||
|
|
||||||
show_color_band(1440, 20, 0x00, 0x00, 0xff, 0x00);
|
|
||||||
|
|
||||||
show_color_band(1440, 20, 0x00, 0x00, 0x00, 0xff);
|
|
||||||
|
|
||||||
show_color_band(1440, 20, 0x00, 0xff, 0xff, 0xff);
|
|
||||||
|
|
||||||
while (1)
|
while (1)
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
@ -1,59 +0,0 @@
|
|||||||
//
|
|
||||||
// Created by longjin on 2022/1/21.
|
|
||||||
//
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#define PAD_ZERO 1 // 0填充
|
|
||||||
#define LEFT 2 // 靠左对齐
|
|
||||||
#define RIGHT 4 // 靠右对齐
|
|
||||||
#define PLUS 8 // 在正数前面显示加号
|
|
||||||
#define SPACE 16
|
|
||||||
#define SPECIAL 32 // 在八进制数前面显示 '0o',在十六进制数前面显示 '0x' 或 '0X'
|
|
||||||
#define SMALL 64 // 十进制以上数字显示小写字母
|
|
||||||
#define SIGN 128 // 显示符号位
|
|
||||||
|
|
||||||
|
|
||||||
#define is_digit(c) ((c) >= '0' && (c) <= '9') // 用来判断是否是数字的宏
|
|
||||||
|
|
||||||
#include "font.h"
|
|
||||||
#include "glib.h"
|
|
||||||
#include <stdarg.h>
|
|
||||||
|
|
||||||
struct screen_info
|
|
||||||
{
|
|
||||||
int width, height; //屏幕大小
|
|
||||||
|
|
||||||
int x, y; //光标位置
|
|
||||||
|
|
||||||
int char_size_x, char_size_y;
|
|
||||||
|
|
||||||
unsigned int *FB_address; //帧缓冲区首地址
|
|
||||||
unsigned long FB_length; // 帧缓冲区长度
|
|
||||||
} pos;
|
|
||||||
|
|
||||||
extern unsigned char font_ascii[256][16]; //导出ascii字体的bitmap(8*16大小)
|
|
||||||
|
|
||||||
char buf[4096]; //vsprintf()的缓冲区
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 将字符串按照fmt和args中的内容进行格式化,然后保存到buf中
|
|
||||||
* @param buf 结果缓冲区
|
|
||||||
* @param fmt 格式化字符串
|
|
||||||
* @param args 内容
|
|
||||||
* @return 最终字符串的长度
|
|
||||||
*/
|
|
||||||
static int vsprintf(char *buf, const char *fmt, va_list args);
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief 将数字按照指定的要求转换成对应的字符串(2~36进制)
|
|
||||||
*
|
|
||||||
* @param str 要返回的字符串
|
|
||||||
* @param num 要打印的数值
|
|
||||||
* @param base 基数
|
|
||||||
* @param field_width 区域宽度
|
|
||||||
* @param precision 精度
|
|
||||||
* @param flags 标志位
|
|
||||||
*/
|
|
||||||
static void write_num(char* str, long long num, int base, int field_width, int precision, int flags);
|
|
Loading…
x
Reference in New Issue
Block a user