单字节的FIFO缓存(30天自制操作系统--读书笔记)

从今天起,写一些读书笔记。最近几个月都在看《30天自制操作系统这本书》,书虽说看的是电子书,但可以花钱买的正版书,既然花费了金钱,就总得有些收获。

任何人都不能总是固步自封,想要进步就得学习别人的知识,对于程序员而言,最简单的方法即是学习别人的代码。

今天的标题是“单字节的FIFO缓存”,其实就是做一个FIFO,看名字就知道了。也就4个函数和1个相关结构体,这样的小代码在嵌入式系统中很常用,也会很好用。

1、相关数据结构体

 struct FIFO8
 {
        unsigned char *buf;     //实际存放数据的内存
        int w_pos,r_pos,size,free_left,flags;
        //w_pos是对缓存写入位置的记录
        //r_pos是对缓存读出位置的记录
        //size是缓存的数据大小
        //free_left是缓存中空出来的可供写入数据的空间大小
        //flags是标志记录
}

2、初始化一个缓存空间

void fifo8_init(struct FIFO8 *fifo,int size,unsigned char *buf)
{
        fifo->buf = buf;
        fifo->size = size;
        fifo->w_pos = 0;
        fifo->r_pos = 0;
        fifo->free_left = size;
        fifo->flags = 0;
} 

3、向缓存中加入一个数据

int fifo8_put(struct FIFO8 *fifo,unsigned char data)
{
        if(fifo->free_left == 0)
       {
               fifo->flags |= FLAG_OVERRUN;      //define FLAG_OVERRUN 1
               return -1;
       }
       //free_left的判断标准就是
       //写人一个数据free_left减一,读出一个数据free_left加一,
       //最开始的时候,free_left为缓存大小,也就是整个缓存都是可以被写入的
       fifo->buf[fifo->w_pos] = data;
       w_pos++;
       if(w_pos == fifo->size)
       {
               w_pos = 0;    //对写指针进行位置限定
       }
       free_left --;
       return 0;
}

4、从缓存中取出一个数据

int fifo8_get(struct FIFO8 *fifo)
{
       int data;
       if(fifo->free_left == fifo->size)
       {
              return  -1;      //整个缓冲区都是空的,也就没有数据可读
        }
        data = fifo->buf[fifo->r_pos];
        fifo->r_pos ++;
        if(fifo->r_pos == fifo->size)
        {
               fifo->r_pos = 0;    //读取位置重定位
         }
         fifo->free_left++;
         return data;                //注意data是unsigned char型,返回值是int型
}

5、获取缓存区的状态 (缓存区有无数据可读)

int fifo8_status(struct FIFO8 *fifo)
{
       return (fifo->size - fifo->free_left);
       //缓存大小减去空着的大小,即缓存中存于数据的个数
}

如何使用上述“单字节的FIFO缓存”代码,这个就很简单了:

1、先初始化

unsigned char buf_for_fifo[32];//定义缓存本身

struct FIFO8 testFifo;              //定义缓存结构体

fifo8_init(&testFifo,32,buf_for_fifo);     //调用此函数即完成了初始化

2、在需要的地方向缓存区中写数据

ret = fifo8_put(&testFifo,1);     //向缓存区中写数据1

3、在需要的地方查询缓存中有无存放数据,并读取出来

if(fifo8_status(&testFifo) > 0)

{

data = fifo8_get(&testFifo);

//处理这些数据即可

}

以上代码设计的要点就是数据结构体设计,另外代码具有良好的可移植性,其关键操作代码中,传入的参数都是其核心数据结构的指针,这一点是不是很像C++中的this指针。

今天到此为止。

时间: 2024-08-12 05:06:13

单字节的FIFO缓存(30天自制操作系统--读书笔记)的相关文章

多定时器处理1(30天自制操作系统--读书笔记)

自认为写过很多MCU程序,但总是回头想想,我所了解的MCU编程思想大体有两种,其中具体的想法我得再找时间写下来. 总想总结出一个可扩展的,易移植的写法,但能力还没到这个层次.但<30天自制操作系统>这本书确实给我了一个思路,就像我已经写过的两篇读书笔记. 将两个独立的内容--FIFO和内存动态管理做到高度模块化,尤其是其中数据结构模型的设计更是我学习的好例子. 今天要学习的设计内容是多定时器处理.原书对这部分的处理讲的很详细,由浅入深,看得我由衷佩服作者,也可能是因为我水平低,稍稍看出点门道来

内存管理(30天自制操作系统--读书笔记)

今天继续读书笔记,“挑战内存管理”(30天自制操作系统). 为什么对这块内容敢兴趣呢,因为曾经遇到这么一个问题.在STM32程序中想使用队列,可不是上篇讲的FIFO,而是使用了较大的内存空间,又想做队列的顺序存取管理. 在这个队列里用到了malloc,动态申请内存,一开始是直接申请不到内存,后来在启动脚本里更改了设置堆的地址值,可以申请成功,但发现申请几次后,也申请不到内存. 果然MCU级别的程序,内存这块处理起来就没有windows程序那么随心所欲了.讲了这么多,开始正题吧. 1.相关数据结构

多定时器处理2(30天自制操作系统 -- 读书笔记)

今天的内容依然来自<30天自制操作系统>这本书. 1.中断处理程序存在的问题,中断处理程序Timer_Interrupt如下: //定时器中断函数 void Timer_Interrupt(void) { int i=0; timerctl.count ++; for(i=0;i<MAX_TIMER;i++) //扫描所有的定时器 { if(timerctl.timer[i].flag == TIMER_FLAG_USING) { timerctl.count--; if(timerct

30天自制操作系统读书笔记(一)

刚开始作者叫我们用二进制编辑器,手敲01代码写了个IMG(磁盘映像文件) ,并提供了一些文件 : Run.bat, install.bat这些一个批处理文件,其实就是写好了cmd指令保存到文档里面,方便以后执行. 其中run.bat里面的指令是这样的 : copy helloos.img ..\z_tools\qemu\fdimage0.bin ..\z_tools\make.exe         -C ../z_tools/qemu (稍微解释一下:第一行:copy指令把,helloos.i

30天自制操作系统读书笔记(五)

为了让程序灵活点,作者觉得把什么320X200这些数据直接写入程序,不如让程序自己获取. 所以就引入了结构体,指针.用指针直接在内存中获取这些数据 (如程序里的这句: Binfo_scrnx = (short *) 0xff4; ). 对于结构体和指针我就不细写了,有C语言基础的人都知道. 因为已经进入了32位模式,所以不能再用bios写好的中断程序给我们输出字符了,要手动了! 用像素点描出图形. 字符可以使用8X16的长方形像素点来表示,转变为16进制就是这样: static char fon

多定时器处理3(30天自制操作系统 -- 读书笔记)

继续定时器中断处理的改进. 1.定时器中断程序Timer_Interrupt是这样的. //定时器中断函数 void Timer_Interrupt(void) { int i=0; timerctl.count ++; for(i=0;i<MAX_TIMER;i++) //扫描所有的定时器 { if(timerctl.timer[i].flag == TIMER_FLAG_USING) { timerctl.count--; if(timerctl.count == 0) //减得时间到了 {

30天自制操作系统读书笔记(二)

我还以为马大哈的作者忘记解释之前那段好长的汇编代码了,留下这么多疑惑! 第二天作者就交代了:以后再讲,第二天我们先来谈谈程序核心部分的内容吧! 好吧,刚准备欢喜的等着作者解释,没想到他就扔了一大串代码(什么鬼?) ; hello-os ; TAB=4 ORG 0x7c00 ; 指明程序的装载地址,这个指令是和nask编译器说的(伪指令)告诉nask,开始执行的时候,把代码放到0x7c00的地方. ;为什么是0x7c00呢,因为规定了0x7c00-0x7dff是启动区的装载地址. ; 以下用于标准

30天自制操作系统读书笔记(四)

想要在画面里画点什么东西,就需要往VRAM的内存区里写入点什么东西. 但是如何写入呢? 作者先用了汇编: _write_mem8:         ; void write_mem8(int addr, int data); MOV                  ECX,[ESP+4]               ; [ESP + 4]中存放的是地址,将其读入ECX MOV                  AL,[ESP+8]                 ; [ESP + 8]中存放的

《30天自制操作系统》笔记(07)——内存管理

<30天自制操作系统>笔记(07)--内存管理 进度回顾 上一篇中处理掉了绝大部分与CPU配置相关的东西.本篇介绍内存管理的思路和算法. 现在想想,从软件工程师的角度看,CPU也只是一个软件而已:它的功能就是加载指令.执行指令和响应中断,而响应中断也是在加载指令.执行指令.就像火车沿着一条环形铁轨前进:当中断发生时,就好像铁轨岔口处变轨了,火车就顺着另一条轨迹走了:走完之后又绕回来重新开始.决定CPU是否变轨的,就是CPU里的特定寄存器. 这是题外话,就此为止. 什么是内存管理 假设内存大小是