🆕 完成了printk(暂不支持浮点数打印)

This commit is contained in:
fslongjin 2022-01-23 23:17:52 +08:00
parent ebb5ef2141
commit ead838bffd
8 changed files with 443 additions and 137 deletions

View File

@ -2,6 +2,8 @@
"files.associations": {
"stdlib.h": "c",
"stdbool.h": "c",
"printk.h": "c"
"printk.h": "c",
"stdarg.h": "c",
"font.h": "c"
}
}

View File

@ -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
objcopy -I elf64-x86-64 -S -R ".eh_frame" -R ".comment" -O binary kernel ../bin/kernel/kernel.bin
kernel: head.o main.o
ld -b elf64-x86-64 -o kernel head.o main.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 AMDs x86-64 architecture.
gcc -mcmodel=large -fno-builtin -m64 -c main.c
kernel: head.o main.o printk.o
ld -b elf64-x86-64 -z muldefs -o kernel head.o main.o printk.o -T link.lds
head.o: head.S
gcc -E head.S > head.s # 预处理
as --64 -o head.o head.s
clean:
rm -rf *.o *.s~ *.s *.S~ *.c~ *.h~ kernel
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 AMDs 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)

View File

@ -4,9 +4,11 @@
//
#pragma once
#ifndef GLIB_H
#define GLIB_H
//引入对bool类型的支持
#include<stdbool.h>
#include <stdbool.h>
#define NULL 0
@ -24,6 +26,8 @@
#define io_lfence() __asm__ __volatile__("lfence\n\t" :: \
: "memory") // 在lfence指令前的读操作当必须在lfence指令后的读操作前完成。
#define ABS(x) ((x) > 0 ? (x) : -(x)) // 绝对值
//链表数据结构
struct List
{
@ -74,20 +78,21 @@ static inline void list_del(struct List *entry)
entry->next = entry->prev;
}
static inline bool list_empty(struct List* entry)
static inline bool list_empty(struct List *entry)
{
/**
* @brief
* @param entry
*/
if(entry->prev == entry->next)
if (entry->prev == entry->next)
return true;
else return false;
else
return false;
}
//计算字符串的长度经过测试该版本比采用repne/scasb汇编的运行速度快16.8%左右)
static inline int strlen(char* s)
static inline int strlen(char *s)
{
register int __res = 0;
while (s[__res] != '\0')
@ -96,3 +101,5 @@ static inline int strlen(char* s)
}
return __res;
}
#endif

View File

@ -3,7 +3,66 @@
//
#include "printk.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)
{
/**
@ -18,6 +77,22 @@ int skip_and_atoi(const char **s)
}
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)
{
/**
@ -53,11 +128,11 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
//清空标志位和field宽度
field_width = flags = 0;
++fmt;
bool flag_tmp = true;
bool flag_break = false;
++fmt;
while (flag_tmp)
{
switch (*fmt)
@ -67,13 +142,7 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
flag_break = true;
flag_tmp = false;
break;
case '%':
//输出 %
*str = '%';
++str;
++fmt;
flag_break = true;
break;
case '-':
// 左对齐
flags |= LEFT;
@ -107,13 +176,21 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
break;
//获取区域宽度
field_width = -1;
if (*fmt == '*')
{
field_width = va_arg(args, int);
++fmt;
}
else if (is_digit(*fmt))
{
field_width = skip_and_atoi(&fmt);
if (field_width < 0)
{
field_width = -field_width;
flags |= LEFT;
}
}
//获取小数精度
precision = -1;
@ -137,29 +214,34 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
qualifier = *fmt;
++fmt;
}
//为了支持lld
if (qualifier == 'l' && *fmt == 'l', *(fmt + 1) == 'd')
++fmt;
//转化成字符串
long long *ip;
switch (*fmt)
{
//输出 %
case '%':
*str++ = '%';
break;
// 显示一个字符
case 'c':
//靠右对齐
if (!(flags & LEFT))
{
while (--field_width)
while (--field_width > 0)
{
*str = ' ';
++str;
}
}
else //靠左对齐
{
*str = (char)va_arg(args, int);
++str;
--field_width;
}
while (--field_width)
*str++ = (unsigned char)va_arg(args, int);
while (--field_width > 0)
{
*str = ' ';
++str;
@ -208,10 +290,13 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
break;
//以八进制显示字符串
case 'o':
flags |= SMALL;
case 'O':
flags |= SPECIAL;
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
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;
//打印指针指向的地址
@ -222,7 +307,7 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
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;
@ -230,34 +315,35 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
case 'x':
flags |= SMALL;
case 'X':
flags |= SPECIAL;
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
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;
//打印十进制有符号整数
case 'i':
case 'd':
case 'ld':
flags |= SIGN;
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
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;
//打印十进制无符号整数
case 'u':
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
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;
//输出有效字符数量到*ip对应的变量
case 'n':
long long *ip;
if (qualifier == 'l')
ip = va_arg(args, long long *);
else
@ -269,18 +355,20 @@ static int vsprintf(char *buf, const char *fmt, va_list args)
//对于不识别的控制符,直接输出
default:
*str++ = '%';
if(*fmt)
if (*fmt)
*str++ = *fmt;
else --fmt;
else
--fmt;
break;
}
}
*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
@ -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)
return 0;
char pad, sign, tmp_num[100];
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';
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;
}
}
@ -351,7 +438,7 @@ static void write_num(char *str, long long num, int base, int field_width, int p
// 靠右对齐
if (!(flags & LEFT))
while (field_width--)
while (field_width-- > 0)
*str++ = pad;
if (sign)
@ -371,11 +458,119 @@ static void write_num(char *str, long long num, int base, int field_width, int p
*str++ = '0';
}
while (js_num--)
while (js_num-- > 0)
*str++ = tmp_num[js_num];
while (field_width--)
while (field_width-- > 0)
*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
View 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字体的bitmap8*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

View File

@ -1,41 +1,70 @@
//
// 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:
* @param val:
/**
* @brief
*
*/
for (int i = 0; i < width * height; ++i)
{
*((char *)address + 0) = d;
*((char *)address + 1) = c;
*((char *)address + 2) = b;
*((char *)address + 3) = a;
++address;
}
printk("\n\n");
for (int i = 0; i < 74; ++i)
printk(" ");
printk_color(0x00e0ebeb, 0x00e0ebeb, " \n");
for (int i = 0; i < 74; ++i)
printk(" ");
printk_color(BLACK, 0x00e0ebeb, " Welcome to DragonOS ! \n");
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)
{
// 初始化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)
;
}

View File

@ -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字体的bitmap8*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);