Linux内核初探
-  linux操作系统历史 
-  开发模式 -  git - 分布式管理
- git clone 获取
- git push 提交
- git pull 更新
 
-  邮件组 mailing list 
-  patch 
 
-  
-  内核代码组成 -  Makfile 
-  arch 体系系统架构相关 
-  block 块设备 
-  crypto 加密算法 
-  drivers 驱动(85%) - atm 通信
- bluetooth 蓝牙
 
-  firmware:外设 
-  fs 文件系统 
-  include 头文件 
-  init 启动代码 
-  ipc 进程通信代码 
-  kernel 内核 
-  lib 库 
-  mm 内存管理 
-  net 网络相关 
-  scripts 脚本 
-  security 安全相关 
-  sound 音频相关 
-  tools 工具(perf 测试工具) 
-  usr 生成打包等 
-  virt 虚拟化 
 
-  
-  内核配置和编译 - 工具链(编译器 arm-linux-gcc、链接工具 ld、查看 objdump、调试)
- 配置内核(决定需要编译的代码模块)
- 编译内核(生成可执行文件)
 
-  vmlinux(原始,未压缩的可执行文件)、zImage(压缩的可执行文件=压缩vmlinux+解压程序头)和uImage(用于给uboot引导的zImage) 
花里胡哨:
ls -a 显示隐藏文件,即.***
 Y 编译内核 N 不编译内核 M 模块编译
 make -j4 zImage
 CC 一个.o ;LD 一个目标下的.o
Linux内核的基本概念
-  物理地址&虚拟地址 -  地址转换:内核使用的地址一定是内核地址,但是cpu使用的是物理地址,中间存在一个转换过程。 - phys_to_virt:
- virt_to_phys:
  
 
-  页表:虚拟 -》物理;内存管理最小单位为页(大小为4k);页表大小=所有物理地址/4k。 - 页表初始化:start_kernel->mm_init->mm_alloc_pgd(划出一段内存,页表初始化)
  
- ioremap(cookie,size):最后调用 分配pgd
 
- 页表初始化:start_kernel->mm_init->mm_alloc_pgd(划出一段内存,页表初始化)
-  内存属性:可读、可写、可执行。 
 
-  
-  Linux内核中的内存管理 -  页(4k byte)为单位进行管理 
-  分配内存举例 - alloc_pages(gfp_mask,order):分配内存的底层调用,可以分配整页的内存;gfp_mask是不同选项的组合(GFP_NOWAIT:分配不允许等待;GFP_ATOMIC;GFP_NOFS:不能进行文件操作;GFP_KERNEL:给内核用;GFP_USER;GFP_IOFS:分配内存可以IO操作);order分配2的次方的页
- kmalloc:分配内存的函数(以字节为单位),分配内存必定是连续的。
- vmalloc:分配内存物理上不连续,虚拟地址连续。
 
 
-  
-  系统调用(Linux特用):用户调用内核函数(API)的媒介,具体实现(通过一个 异常 ,使得应用程序陷入内核中执行) 
-  内核进程、线程 -  内核只有线程,没有进程 
-  线程的task_struct(核心结构):包括进程的优先级、堆栈信息、打开文件句柄信息 
  
  
-  线程的调度(3种优先级:SCHED_RR、SCHED_PRI、SCHED_RT);调度的时机:时钟中断发生;Linux发生内核态和用户态相互切换;Liunx执行完信号 
 
-  
-  内核中同步和线程间通信方式 -  原子操作:不可能被打断的基本操作 
-  同步通信方式 -  自旋锁(spin_lock) 
  
-  信号量(up:释放 和 down:) 
  
  
 
-  
-  异步通信方式 - 信号 – signal :处理信号的函数在task_struct中指明。
  
 
- 信号 – signal :处理信号的函数在task_struct中指明。
 
-  
-  中断 - 中断的硬件概念:就是一个外部的电平信号
- 中断处理的上半部:需要linux内核关闭其他硬件信号
- 中断处理的下半部:上半部中来不及处理、比较冗长的程序段;需要等待其他程序结果,或者需要等待获取其他资源的程序段。
 
-  时钟和定时器管理 - 时钟硬件概念:能够产生一种定时中断的电路
- RTC(实时时钟或实时计数器)和system timer(很多情况就是用来作为延迟和计算相对时间)
- tick(时钟中断的周期,HZ)& jiffies(全局变量,如果是64位系统,就是64位的变量,记录了从上电开始,所经历的tick数)
 
-  文件系统 - 虚拟文件系统(VFS):是linux内核为了屏蔽物理文件系统的差异所产生的一个中间层。
- 物理文件系统(ext4:最大特点是具有完善的日志系统,yaffs2:比较适合在nand flash部署的文件系统,ubifs和btfs)
 
Linux驱动程序开发基础
-  内核模块编程 -  驱动模块化编程的好处 - 驱动编译进内核,导致内核非常大
- 很多驱动都只是在特定机器上使用
- 实现热插拔提供基础
 
-  模块编程的代码实例 - 实例验证hello.ko模块
  
  
 
- 实例验证hello.ko模块
-  Makefile的写法 
 
-  

- 其他
	- 模块位置可以随意放置
	- 内核代码更新,模块代码也要更新
-  驱动程序访问硬件的特殊性 -  DMA -  dma的基本硬件概念:是存在外设设备中的一个硬件控制器,作用是不需要cpu协助,就可以搬移内存数据到外设的存储设备中。 
-  dma的基本配置过程:通过程序配置dma控制器,告诉dma控制器它可以访问的内存地址。然后cpu将要传给外设的数据写到事前约定好的地址。 
 
-  
-  IO子系统:在嵌入式系统中,实现对外围附属设备进行控制的有效手段。通过IO端口进行0和1操作,可以发指令或者传递信息给附属设备。 
 
-  
-  Linux设备模型:最初目的是实现只能电源管理 -  kobject -  一个设备驱动会建立一个kobject,偶尔也有因为功能复杂的原因而建立多个kobject。 
-  koject会有一个一个kobj_type属性。 
  - entry :本人所在目录 - parent:父节点 - kset:一组类似性质的kobject集合 - ktype:类型 - kref:这个对象的引用计数
-  kobject一般都是sysfs中的一个目录,从而形成对用户空间的交互。 
-  引用计数kref。 
 
-  
-  kset:一组类似性质的kobject集合;一个kset也就是一个子系统,它是sysfs的一个顶层目录的表征。比如block子系统,各种总线子系统。 
-  sysfs:虚拟文件系统(管理内核的设备而非磁盘,它是kobject对象的完整视图) -  主要api(sysfs_create_file; sysfs_create_link; sysfs_remove_file; sysfs_remove_link) 
-  提供丰富的内核和用户空间交互的手段 
-  sysfs主要目录结构 - block:块设备,独立于所链接的总线。 - devices:被所有内核识别的硬件设备,依照链接他们的总线对其进行组织。 - bus:系统中用于连接设备的总线 - drivers:在内核中注册的设备驱动程序 - class:系统中设备的类型(声卡,网卡,显卡等);同一类可能包含由不同总线连接的设备,于是由不同的驱动程序驱动。 - power:处理一些硬件设备电源状态的文件。 - firmware:处理一些硬件设备固件的文件。
 
-  
-  udev:处理热插拔机制,通过设备驱动加载时,注册kobject后,向用户空间发送uevent实现的。 
 
-  
-  Linux驱动的分类 -  字符设备:一般都是以串行顺序依次进行访问,典型的包括触摸屏,鼠标,按键等。 - cdev结构:
  
 
- cdev结构:
-  块设备:一般以扇区、块为单位进行读写访问,例如硬盘,cdrom,flash等 
  
-  网络设备:以太网的设备。 
-  杂项设备:没法归类或复合设备。 
 
-  
-  Linux内核的基本调试方法 -  printk - 级别
- dmesg
- 打开和关闭调试信息
 
-  oops 
-  kprobe 
-  kcore 
 
-  



















