MacOS环境-手写操作系统-18-图片分层技术

简介: MacOS环境-手写操作系统-18-图片分层技术

图片分层技术

1.简介

在前面的章节中 我们完成了鼠标的绘制 但存在一个严重的问题是


当鼠标移动的时候 鼠标下方的窗口或图案会因为内核重绘鼠标而被擦除掉


解决这个问题的办法是 每当鼠标移动时 我们先把底下的图案绘制一遍


然后把鼠标在移动后的新坐标处再绘制一遍


由此 我们引出一个叫图层的概念 鼠标下面的背景是一个图层 鼠标自己处于一个图层 图层间有高度差别


作为背景的图层 它的高度就要比鼠标所在的图层低 因此内核每次重绘时 先绘制高度低的图层 再绘制高度高的图层


我们使用的操作系统 用鼠标挪动窗口时 窗口底部的图案不受影响 就是使用了图层分层技术


基本原理如下图的:

2.代码

有了基本原理后,我们可以编码实现了,首先定义图层的数据结构


struct SHEET {
    unsigned char *buf;
    int bxsize, bysize, vx0, vy0, col_inv, height, flags;
};


buf对应的是窗口的像素信息


bxsize, bysize 是窗口的长和高


vx0, vy0是窗口移动后的左上角坐标


col_inv 用来表示窗口中不显示的部分


height表示图层的高度


flags用来表示图层状态


这里需要解释一下的变量是col_inv


我们看看用来表示鼠标的buf:

static char cursor[16][16] = {
        "**************..",
        "*OOOOOOOOOOO*...",
        "*OOOOOOOOOO*....",
        "*OOOOOOOOO*.....",
        "*OOOOOOOO*......",
        "*OOOOOOO*.......",
        "*OOOOOOO*.......",
        "*OOOOOOOO*......",
        "*OOOO**OOO*.....",
        "*OOO*..*OOO*....",
        "*OO*....*OOO*...",
        "*O*......*OOO*..",
        "**........*OOO*.",
        "*..........*OOO*",
        "............*OO*",
        ".............***"
    };

上面的数组对应的就是SHEET结构中的buf, 我们绘制鼠标时


是按照数组中的元素对应到像素点的


如果某个元素的值是*那么对应的像素点是黑色


如果元素的值是0 那么对应像素的颜色就是白色


如果元素的值是 . 那么这些点对应的像素颜色不做任何改变


这样的话 下面图层的相关图案就可以“透视”出来


这些点对应的值就是SHEET中的col_inv


有了图层后,我们还要定义图层管理器对象


#define MAX_SHEETS  256
struct SHTCTL {
    unsigned char *vram;
    int xsize, ysize, top;
    struct SHEET *sheets[MAX_SHEETS];
    struct SHEET sheets0[MAX_SHEETS];
};
#define SIZE_OF_SHEET  32
#define SIZE_OF_SHTCTL 9232


其中vram 对应的是显存地址 也就是0xa0000, xsize, ysize 代表整个显示界面的宽和高


当前系统显示界面的大小是320 * 320


top 表示当前要显示几个图层


sheets0 是用来存储图层对象的数组


sheets 是指针数组 用来指向下面图层数组中的对应图层对象


根据图层数据结构 定义用来操作他们的相关函数接口:


所有代码都定义在文件win_sheet.h中


struct SHEET *sheet_alloc(struct SHTCTL *ctl);
struct SHTCTL *shtctl_init(struct MEMMAN *memman, unsigned char *vram,
  int xsize, int ysize);
void sheet_setbuf(struct SHEET *sht, unsigned char *buf, int xsize, int ysize,
    int col_inv);
void sheet_updown(struct SHTCTL *ctl, struct SHEET *sht, int height);
int sheet_refresh(struct SHTCTL *ctl);
void sheet_slide(struct SHTCTL *ctl, struct SHEET *sht, int vx0, int vy0);


我们继续看看相关代码的实现

#include "mem_util.h"
#include "win_sheet.h"

struct SHTCTL *shtctl_init(struct MEMMAN *memman, unsigned char *vram,
  int xsize, int ysize) {
    struct SHTCTL *ctl;
    int i;
    ctl = (struct SHTCTL *)memman_alloc_4k(memman, SIZE_OF_SHTCTL);
    if (ctl == 0) {
        return 0;
    }

    ctl->vram = vram;
    ctl->xsize = xsize;
    ctl->ysize = ysize;
    ctl->top = -1;
    for (i = 0; i < MAX_SHEETS; i++) {
        ctl->sheets0[i].flags = 0;
    }

    return ctl;
}

#define SHEET_USE  1
struct SHEET *sheet_alloc(struct SHTCTL *ctl) {
    struct SHEET *sht;
    int i;
    for (i = 0; i < MAX_SHEETS; i++) {
        if (ctl->sheets0[i].flags == 0) {
            sht = &ctl->sheets0[i];
            ctl->sheets[i] = sht;
            sht->flags = SHEET_USE;
            sht->height = -1;
            return sht;
        }
    }

    return 0;
}

上面是两个初始化函数 分别初始化图层管理器和图层对象


在这两个函数中 使用到一个来自内存管理器的接口memman_alloc_4k


顾名思义 就是向管理器申请一块大小为4k整数倍的内存


即使你只使用1字节 分配到的内存大小仍然是4k


一下子分配4k是为了防止内存碎片化过于严重 以造成浪费


我们看看memman_alloc_4k的实现,该函数的代码在文件mem_util.c中:


unsigned int memman_alloc_4k(struct MEMMAN *man, unsigned int size) {
    unsigned int a;
    size = (size + 0xfff) & 0xfffff000;
    a = memman_alloc(man, size);
    return a;
}


它的实现逻辑是这样的 4k对应的16进制数是0x1000


如果你申请大小是1字节 那么 size + 0xfff 就等于是 0x1000


跟0xfffff000 做与操作后 结果任然是0x1000


如果你申请的内存大小是 0xfff 也就是差1字节到4k 那么size + 0xfff的结果是0x1FFE


做与运算后结果也是0x1000


所以这种计算方法能够在你申请任何大小内存的情况下 将大小圆整到大于所申请内存的最小4k倍数


sheet_alloc用来分配一个图层对象给指定窗口 同时我们把图层的高设置成-1时 表示图层对应的窗口处于最小化状态


void sheet_setbuf(struct SHEET *sht, unsigned char *buf, int xsize, int ysize,
    int col_inv) {
    sht->buf = buf;
    sht->bxsize = xsize;
    sht->bysize = ysize;
    sht->col_inv = col_inv;
    return;
}


sheet_setbuf将窗口对应的二位数组与图层对象关联起来


例如上面的数组cursor就是鼠标绘制数值 cursor数组将会跟通过上面的函数跟其图层对象关联起来


下面我们看的将是最为复杂的一个函数

void sheet_updown(struct SHTCTL *ctl, struct SHEET *sht, int height) {
    int h, old = sht->height;
    if (height > ctl->top + 1) {
        height = ctl->top + 1;
    }

    if (height < -1) {
        height = -1;
    }

    sht->height = height;

    if (old > height) {
        if (height >= 0) {
            for (h = old; h > height; h--) {
                ctl->sheets[h] = ctl->sheets[h-1];
                ctl->sheets[h]->height = h;
            }

            ctl->sheets[height] = sht;
        } else {
            if (ctl->top > old) {
               for (h = old; h < ctl->top; h++) {
                   ctl->sheets[h] = ctl->sheets[h+1];
                   ctl->sheets[h]->height = h;
               }
            }

            ctl->top--;
        }

        sheet_refresh(ctl);
    } else if (old < height) {
        if (old >= 0) {
            for (h = old; h < height; h++) {
               ctl->sheets[h] = ctl->sheets[h + 1];
               ctl->sheets[h]->height = h;
            }

            ctl->sheets[height] = sht;
        } else {
            for (h = ctl->top; h >= height; h--) {
                ctl->sheets[h + 1] = ctl->sheets[h];
                ctl->sheets[h + 1]->height = h + 1;
            }

            ctl->sheets[height] = sht;
            ctl->top++;
        }

        sheet_refresh(ctl);
    }

}

这个函数比较大 我们分块分析


总体来说sheet_updown是用来调整每个窗口图层的高度的


某个窗口被点击后 它会从底部冒到顶部 也就是原来被遮住看不到的部分会全部显示出来


这种情况就是up 如果窗口被其他拖过来的窗口给遮住 这种情况就是down


第一部分代码如下


int h, old = sht->height;
    if (height > ctl->top + 1) {
        height = ctl->top + 1;
    }
    if (height < -1) {
        height = -1;
    }
    sht->height = height;


如果是up 也就是传进来的参数height比图层对应的高度要高 那么就改变图层的高度值


如果调整的高度过大 比如当前只桌面上只有三个窗口(ctl->top == 3)


但你要把新增窗口设置成100 那么前面这段代码会把当前窗口的高度设置为4


如果窗口下降高度,例如当前有三个窗口图层:


sheet1{height = 1;}
sheet2{height = 2;}
sheet3{hegiht = 3;}


现在最高的窗口sheet3高度要降为最低,那么就把sheet1, sheet2分别往后挪一个位置,然后将sheet3 放到最上面:


sheet3{hegiht = 1;}
sheet1{height = 2;}
sheet2{height = 3;}


对应的代码


if (old > height) {
        if (height >= 0) {
            for (h = old; h > height; h--) {
                ctl->sheets[h] = ctl->sheets[h-1];
                ctl->sheets[h]->height = h;
            }
            ctl->sheets[height] = sht;
        } else {
            ....
        }
    sheet_refresh(ctl);
 }


如果当前窗口最小话 那么该窗口对应的图层对象就从图层数组中去掉


例如sheet3的窗口最小化 那么图层数组由原来的


sheet1{height = 1;}
sheet2{height = 2;}
sheet3{hegiht = 3;}


变为


sheet1{height = 1;}
sheet2{height = 2;}


对应的代码


if (old > height) {
        if (height >= 0) {
           。。。。
        } else {
            if (ctl->top > old) {
               for (h = old; h < ctl->top; h++) {
                   ctl->sheets[h] = ctl->sheets[h+1];
                   ctl->sheets[h]->height = h;
               }
            }
            ctl->top--;
        }
    sheet_refresh(ctl);
 }


sheet_refresh(ctl); 的作用是根据图层数组绘制图层,后面再详细分析


如果当窗口的高度变高 底层的窗口激活后要在最上头显示


例如有下面三个图层对应的窗口


sheet1{height = 1;}
sheet2{height = 2;}
sheet3{hegiht = 3;}



如果sheet1对应的窗口从最低位置转换到最前端显示,那么图层数组就做如下变得


sheet2{height = 1;}
sheet3{hegiht = 2;}
sheet1{height = 3;}


对应的逻辑代码


else if (old < height) {
        if (old >= 0) {
            for (h = old; h < height; h++) {
               ctl->sheets[h] = ctl->sheets[h + 1];
               ctl->sheets[h]->height = h;
            }
            ctl->sheets[height] = sht;
        } else {
            。。。。
        }
        sheet_refresh(ctl);
    }


如果某个窗口原来处于最小化 现在要把它的高度设置为2


那么原来高度大于2的窗口 他们的高度就要相应的增加


然后再把空出来的图层位置让给新改变的图层


例如当前有三个窗口对应的图层如下


sheet1{height = 1;}
sheet2{height = 2;}
sheet3{hegiht = 3;}


新窗口对应的图层是sheet4,它的高度从-1


也就是最小化形态转为高度为2


那么就需要把原来高度为2以上的图层在图层数组中往后挪


并改变高度


sheet1{height = 1;}
sheet2{height = 3;}
sheet3{hegiht = 4;}


再把sheet4图层插入挪动后空出来的位置


sheet1{height = 1;}
sheet4{height = 2;}
sheet2{height = 3;}
sheet3{hegiht = 4;}


对应的逻辑代码


else if (old < height) {
        if (old >= 0) {
           。。。。。
        } else {
            for (h = ctl->top; h >= height; h--) {
                ctl->sheets[h + 1] = ctl->sheets[h];
                ctl->sheets[h + 1]->height = h + 1;
            }
            ctl->sheets[height] = sht;
            ctl->top++;
        }
        sheet_refresh(ctl);
    }


图层的绘制函数

int sheet_refresh(struct SHTCTL *ctl) {
    int h, bx, by, vx, vy;
    unsigned char *buf, c , *vram = ctl->vram;
    struct SHEET *sht;
    for (h = 0; h <= ctl->top; h++) {

        sht = ctl->sheets[h];
        buf = sht->buf;
        for (by = 0; by < sht->bysize; by++) {

            vy = sht->vy0 + by;
            for (bx = 0; bx < sht->bxsize; bx++) {

                vx = sht->vx0 + bx;
                c = buf[by * sht->bxsize + bx];
                if (c != sht->col_inv) {
                    vram[vy * ctl->xsize + vx] = c;
                }
            }
        }
    }

    return 0;
}

3.编译运行(这次比上次要多一些)

由于模块多起来了 而且是分开的


因此它们需要单独编译 然后再把这些的 .o 文件合并成一个模块


编译过程如下:


(我是在MAC下 当前版本是 MacOS BigSur)


先使用命令编译mem_util.c


再编译write_vga_desktop.c


i386-elf-gcc -m32 -fno-asynchronous-unwind-tables -s -c -o mem_util.o mem_util.c
i386-elf-gcc -m32 -fno-asynchronous-unwind-tables -s -c -o win_sheet.o win_sheet.c
i386-elf-gcc -m32 -fno-asynchronous-unwind-tables -s -c -o write_vga_desktop.o write_vga_desktop.c


编译后 用ld将其合并链接起来


i386-elf-ld -m elf_i386 -r write_vga_desktop.o mem_util.o win_sheet.o -o ckernel.o


链接后 对其进行反编译


./objconv -fnasm ckernel.o ckernel.asm


接着删除其中多余的部分


(好几千行我就不放出来了)


接着再配合kernel一起汇编


nasm -o kernel.bat kernel.asm


(如果出现错误 说跳转不能太长 记得添加 near 我当前环境没有报错 所以我就不处理了)


运行java 生成 system.img


目录
相关文章
|
3月前
|
存储 安全 物联网
操作系统的心脏:深入理解现代操作系统架构与核心技术
本文旨在为读者提供一个关于现代操作系统(OS)架构和核心技术的全面概述。通过分析OS的主要组件、功能以及它们如何协同工作,本文揭示了操作系统在计算机系统中的核心地位及其复杂性。我们将探讨进程管理、内存管理、文件系统和输入/输出(I/O)等关键技术,并讨论它们对系统性能的影响。此外,本文还将涵盖一些最新的操作系统趋势和技术,如云计算、虚拟化和物联网(IoT)。通过阅读本文,读者将获得对操作系统内部运作方式的深刻理解,这对于软件开发人员、IT专业人士以及对计算机科学感兴趣的任何人来说都是宝贵的知识。
|
18天前
|
Unix Linux Docker
CentOS停更沉寂,RHEL巨变限制源代:Docker容器化技术的兴起助力操作系统新格局
操作系统是计算机系统的核心软件,管理和控制硬件与软件资源,为用户和应用程序提供高效、安全的运行环境。Linux作为开源、跨平台的操作系统,具有高度可定制性、稳定性和安全性,广泛应用于服务器、云计算、物联网等领域。其发展得益于庞大的社区支持,多种发行版如Ubuntu、Debian、Fedora等满足不同需求。
44 4
|
2月前
|
机器学习/深度学习 人工智能 Android开发
移动应用开发与操作系统的协同进化:探索现代技术融合之道###
随着移动互联网的迅猛发展,移动应用已成为人们日常生活中不可或缺的一部分。本文深入探讨了移动应用开发的最新趋势、关键技术以及移动操作系统的发展如何相互促进,共同推动移动互联网的创新与变革。通过分析当前市场动态和技术挑战,本文旨在为开发者提供有价值的见解和指导,帮助他们在竞争激烈的市场中脱颖而出。
|
3月前
|
监控 Linux 云计算
Linux操作系统在云计算环境中的实践与优化###
【10月更文挑战第16天】 本文探讨了Linux操作系统在云计算环境中的应用实践,重点分析了其在稳定性、安全性和高效性方面的优势。通过具体案例,阐述了Linux如何支持虚拟化技术、实现资源高效分配以及与其他开源技术的无缝集成。文章还提供了针对Linux系统在云计算中的优化建议,包括内核参数调整、文件系统选择和性能监控工具的应用,旨在帮助读者更好地理解和应用Linux于云计算场景。 ###
70 3
|
3月前
|
安全 Android开发 iOS开发
安卓vs iOS:探索两种操作系统的独特魅力与技术深度###
【10月更文挑战第16天】 本文旨在深入浅出地探讨安卓(Android)与iOS这两种主流移动操作系统的特色、优势及背后的技术理念。通过对比分析,揭示它们各自如何塑造了移动互联网的生态,并为用户提供丰富多彩的智能体验。无论您是科技爱好者还是普通用户,都能从这篇文章中感受到技术创新带来的无限可能。 ###
72 2
|
2月前
|
人工智能 Anolis 开发者
|
3月前
|
存储 C语言 iOS开发
MacOS环境-手写操作系统-48-让内核从错误中恢复
MacOS环境-手写操作系统-48-让内核从错误中恢复
60 0
|
3月前
|
存储 API C语言
MacOS环境-手写操作系统-46,47-C语言开发应用程序
MacOS环境-手写操作系统-46,47-C语言开发应用程序
45 0
|
3月前
|
编译器 API C语言
MacOS环境-手写操作系统-45-C语言开发应用程序
MacOS环境-手写操作系统-45-C语言开发应用程序
61 0
|
3月前
|
小程序 iOS开发 MacOS
MacOS环境-手写操作系统-44-运行简单的程序
MacOS环境-手写操作系统-44-运行简单的程序
35 0