转了一圈,今天再次回到C

网上一篇博文,个人感觉良心作品,故而拿来重新实现一遍,原作者原文有问题,我这里把他打通了

一.GCC Makefile

//hello.c

#include <stdio.h>

#include <stdlib.h>

void main(void)

{

#ifdef DEBUG

printf("you ask for debug!\n");

#endif

printf("we must say goodbye\n");

return;

}

//Makefile

ifeq ($(DEBUG),y)

CFLAGS := $(CFLAGS) -DDEBUG

endif

hello: hello.c

$(CC) $(CFLAGS) $< -o $@

//result

# make

cc  hello.c -o hello

# ./hello

we must say goodbye

# rm hello

# make DEBUG:=y

cc  -DDEBUG hello.c -o hello

# ./hello

you ask for debug!

we must say goodbye

二.KBuild(kernel gcc) Makefile

//ville.c

/**

* * Author : ville lee villelee1987@gmail.com

* * Create Time : 2010/07/31

* * Description : This module implements a virtual file system.

* * Aimming at learning the theory of Linux VFS.

* *

* * Change Log :

* *    version author  Date    Log

* *

* *

* *

* * */

#include <linux/module.h>

#include <linux/init.h>

#include <linux/kernel.h>

/**

* * init function of the module

* *

* * */

static int __init

ville_init(void)

{

#ifdef DEBUG

/* we give the debug message like: module name : function name : debug message */

printk(KERN_ALERT "ville :ville_init: you ask for debug!\n");

#endif

printk(KERN_ALERT "ville :ville_init: ville module init!\n");

return 0;

}

static void __exit

ville_exit(void)

{

#ifdef DEBUG

/* we give the debug message like: module name : function name : debug message */

printk(KERN_ALERT "ville :ville_exit: you ask for debug!\n");

#endif

printk(KERN_ALERT "ville :ville_exit: ville module exit!\n");

return;

}

module_init(ville_init);

module_exit(ville_exit);

MODULE_LICENSE("GPL");

MODULE_AUTHOR("ville lee");

//Makefile

## if module is built in the kernel module

## system. Give the work to kbuild.

# provide the default value to module name and ccflags-y

ifeq ($(MODULE),)

MODULE := ville

endif

ifeq ($(DEBUG),y)

ccflags-y += -DDEBUG

endif

ifneq ($(KERNELRELEASE),)

obj-m := $(MODULE).o

else

KERNELDIR := /lib/modules/$(shell uname -r)/build

PWD := $(shell pwd)

default:

$(MAKE) -C $(KERNELDIR) M=$(PWD) modules

endif

.PHONY:clean

clean:

echo "cleaning ...."

rm modules.order Module.symvers \

$(MODULE).ko $(MODULE).mod.c \

$(MODULE).mod.o $(MODULE).o

echo "clean up"

//result

# make

make -C /lib/modules/4.13.0-17-generic/build M=/home/woodzcl/c_develop/KMF modules

make[1]: Entering directory '/usr/src/linux-headers-4.13.0-17-generic'

CC [M]  /home/woodzcl/c_develop/KMF/ville.o

Building modules, stage 2.

MODPOST 1 modules

CC      /home/woodzcl/c_develop/KMF/ville.mod.o

LD [M]  /home/woodzcl/c_develop/KMF/ville.ko

make[1]: Leaving directory '/usr/src/linux-headers-4.13.0-17-generic'

# insmod ville.ko

# rmmod ville

# cat /var/log/kern.log

Nov 22 22:05:35 ubuntu kernel: [ 1495.967374] ville :ville_init: ville module init!

Nov 22 22:05:39 ubuntu kernel: [ 1500.103238] ville :ville_exit: ville module exit!

# make clean

echo "cleaning ...."

cleaning ....

rm modules.order Module.symvers \

ville.ko ville.mod.c \

ville.mod.o ville.o

echo "clean up"

clean up

# make DEBUG:=y

make -C /lib/modules/4.13.0-17-generic/build M=/home/woodzcl/c_develop/KMF modules

make[1]: Entering directory '/usr/src/linux-headers-4.13.0-17-generic'

CC [M]  /home/woodzcl/c_develop/KMF/ville.o

Building modules, stage 2.

MODPOST 1 modules

CC      /home/woodzcl/c_develop/KMF/ville.mod.o

LD [M]  /home/woodzcl/c_develop/KMF/ville.ko

make[1]: Leaving directory '/usr/src/linux-headers-4.13.0-17-generic'

# insmod ville.ko

# rmmod ville

# cat /var/log/kern.log

Nov 22 22:08:40 ubuntu kernel: [ 1680.874360] ville :ville_init: you ask for debug!

Nov 22 22:08:40 ubuntu kernel: [ 1680.874362] ville :ville_init: ville module init!

Nov 22 22:08:44 ubuntu kernel: [ 1685.649839] ville :ville_exit: you ask for debug!

Nov 22 22:08:44 ubuntu kernel: [ 1685.649841] ville :ville_exit: ville module exit!

Finally:

最后,发现自己还是个C程序员而已。

哎,但是改变还是要继续,时代在前进,我们每一个老东西都不能落后啊:)哈哈哈哈哈哈哈哈哈哈哈哈

GCC 用户态&内核态 Makefile的更多相关文章

  1. 42.Linux应用调试-初步制作系统调用(用户态->内核态)

    1首先来讲讲应用程序如何实现系统调用(用户态->内核态)? 我们以应用程序的write()函数为例: 1)首先用户态的write()函数会进入glibc库,里面会将write()转换为swi(S ...

  2. linux 用户态 内核态

    http://blog.chinaunix.net/uid-1829236-id-3182279.html 究竟什么是用户态,什么是内核态,这两个基本概念以前一直理解得不是很清楚,根本原因个人觉得是在 ...

  3. 进程:linux用户态-内核态

    用户态:Ring3运行于用户态的代码则要受到处理器的诸多检查,它们只能访问映射其地址空间的页表项中规定的在用户态下可访问页面的虚拟地址,且只能对任务状态段(TSS)中I/O许可位图(I/O Permi ...

  4. 用户态和内核态&操作系统

    用户态和内核态 内核态:cpu可以访问内存的所有数据,包括外围设备,例如硬盘,网卡,cpu也可以将自己从一个程序切换到另一个程序. 用户态:只能受限的访问内存,且不允许访问外围设备,占用cpu的能力被 ...

  5. cpu内核态与用户态

    1.操作系统需要两种CPU状态 内核态(Kernel Mode):运行操作系统程序,操作硬件 用户态(User Mode):运行用户程序 2.指令划分 特权指令:只能由操作系统使用.用户程序不能使用的 ...

  6. Linux用户态和内核态

    究竟什么是用户态,什么是内核态,这两个基本概念以前一直理解得不是很清楚,根本原因个人觉得是在于因为大部分时候我们在写程序时关注的重点和着眼的角度放在了实现的功能和代码的逻辑性上,先看一个例子: 1)例 ...

  7. Linux探秘之用户态与内核态

    一. Unix/Linux的体系架构 如上图所示,从宏观上来看,Linux操作系统的体系架构分为用户态和内核态(或者用户空间和内核).内核从本质上看是一种软件——控制计算机的硬件资源,并提供上层应用程 ...

  8. Linux 用户态和内核态

    1.特权级特权级用来管理和控制程序执行.如Intel x86架构的CPU,有0~3四个特权级,0级最高,3级最低.硬件在执行每条指令时都会检查指令具有的特权级.硬件提供了特权级使用机制,对操作系统来说 ...

  9. 【转载】 Linux用户态和内核态

    [说明]转载自 http://my.oschina.net/liubin/blog/27795 究竟什么是用户态,什么是内核态,这两个基本概念以前一直理解得不是很清楚,根本原因个人觉得是在于因为大部分 ...

随机推荐

  1. aspectj 注解

    aspectj是一个面向切面编程的框架,即实现了aop,这不是spring,它本身很小,方便简洁,spring将其整合成自己的. 与spring本身对aop的支持不同,顾问采用正则表达式或者方法名或通 ...

  2. 洛谷P1316 丢瓶盖【二分】【贪心】

    题目:https://www.luogu.org/problemnew/show/P1316 题意: 给定a个点的坐标(在一条直线上),现在要选b个点,问这b个点的最近距离的最大值是多少. 思路: 感 ...

  3. Mac Mini2018 开箱(视频)全球首映!Kindle 开箱一并奉上(文字)

    2018.12.9更新 为了方便大家收看我的这个开箱视频,我把视频放在知乎上了,可以参考如下连接即可观看啦: https://zhuanlan.zhihu.com/p/51677467 ------- ...

  4. [qemu][kvm] 在一个vmware虚拟机里安装qemu-kvm虚拟机

    说起来这个需求,简直是傻傻的.但却实实在在的摆在我的面前.... VM无外乎就是为了模拟场景:我现在要的场景就是一台很多个core的linux主机.但是我只有一个装了windows的笔记本.上边有一个 ...

  5. [hyperscan] hyperscan 1到1.5 --!!

    [hyperscan][pkg-config] hyperscan 从0到1路线图 接续前文,继续深入理解: 概述: 1.  自动机理论,是hyperscan的理论基础. https://zh.wik ...

  6. [daily][mirror][daemonlogger][tc] 我想把一个网卡(port A)的流量,镜像到虚拟机的一个网卡(port VA)上去

    iptables tee 模块 https://blog.gnuers.org/?p=740 http://blog.csdn.net/wesleyflagon/article/details/385 ...

  7. 图->存储结构->数组表示法(邻接矩阵)

    文字描述 用两个数组分别存储顶点信息和边/弧信息. 示意图 算法分析 构造一个采用邻接矩阵作存储结构.具有n个顶点和e条边的无向网(图)G的时间复杂度是(n*n + e*n), 其中对邻接矩阵G.ar ...

  8. django--验证码功能实现

    首先建立验证码的视图函数1需要安装pillow库 #导入绘图库 from PIL import ImageDraw #导入绘图字体库 from PIL import ImageFont #导入图片库 ...

  9. 20165336 2017-2018-2 《Java程序设计》第2周学习总结

    学号 2017-2018-2 20165336 <Java程序设计>第2周学习总结 教材学习内容总结 第二章 标识符第一个字符不能是数字 标识符不能是关键字 byte型变量的取值范围是-2 ...

  10. java 线程(四)线程安全 同步方法

    package cn.sasa.demo2; import java.util.concurrent.ExecutionException; public class ThreadDemo { pub ...