Linux內核信號SIGIO使用實例講解

2021-12-23 一口Linux
一、信號1. 基本概念

信號是在軟體層次上對中斷機制的一種模擬,在原理上,一個進程收到一個信號與處理器收到一個中斷請求可以說是一樣的。信號是異步的,一個進程不必通過任何操作來等待信號的到達,事實上,進程也不知道信號到底什麼時候到達。

例如鍵盤輸入中斷按鍵(^C),它的發生在程序執行過程中是不可預測的。

信號是進程間通信機制中唯一的異步通信機制,可以看作是異步通知,通知接收信號的進程有哪些事情發生了。

硬體異常也能產生信號,例如被零除、無效內存引用(test裡產生的就是這種錯誤)等。這些條件通常先由內核硬體檢測到,然後通知內核。內核將決定產生什麼樣的信號。

同一個信號的額外發生通常不會被排隊。如果信號在被阻塞時發生了5次,當我們反阻塞這個信號時,這個信號的信號處理函數通常只被調用一次。

同一時刻只能處理一個信號,在信號處理函數發信號給自己時,該信號會被pending。

信號的數值越小,則優先級越高。當進程收到多個待處理信號時,總是先處理優先級別高的信號。

信號處理函數的棧可以使用被中斷的也可以使用獨立的,具體可以通過系統調用設置。

信號機制經過POSIX實時擴展後,功能更加強大,除了基本通知功能外,還可以傳遞附加信息。

2. 處理方式

忽略:接收到信號後不做任何反應。捕獲:用自定義的信號處理函數來執行特定的動作。默認:接收到信號後按系統默認的行為處理該信號。這是多數應用採取的處理方式。

二、Linux下的信號類型

使用kill -l就會顯示出linux支持的信號列表。
其中列表中,編號為1 ~ 31的信號為傳統UNIX支持的信號,是不可靠信號(非實時的),編號為32 ~ 63的信號是後來擴充的,稱做可靠信號(實時信號)。不可靠信號和可靠信號的區別在於前者不支持排隊,可能會造成信號丟失,而後者不會。

下面我們對編號小於SIGRTMIN的信號進行討論(下面的編號 依次對應信號 的數值為1 - 31)。

1) SIGHUP

本信號在用戶終端連接(正常或非正常)結束時發出, 通常是在終端的控制進程結束時, 通知同一session內的各個作業, 這時它們與控制終端不再關聯。

登錄Linux時,系統會分配給登錄用戶一個終端(Session)。在這個終端運行的所有程序,包括前臺進程組和後臺進程組,一般都 屬於這個 Session。當用戶退出Linux登錄時,前臺進程組和後臺有對終端輸出的進程將會收到SIGHUP信號。這個信號的默認操作為終止進程,因此前臺進 程組和後臺有終端輸出的進程就會中止。不過可以捕獲這個信號,比如wget能捕獲SIGHUP信號,並忽略它,這樣就算退出了Linux登錄,wget也 能繼續下載。

此外,對於與終端脫離關係的守護進程,這個信號用於通知它重新讀取配置文件。

2) SIGINT

程序終止(interrupt)信號, 在用戶鍵入INTR字符(通常是Ctrl-C)時發出,用於通知前臺進程組終止進程。

3) SIGQUIT

和SIGINT類似, 但由QUIT字符(通常是Ctrl-)來控制. 進程在因收到SIGQUIT退出時會產生core文件, 在這個意義上類似於一個程序錯誤信號。

4) SIGILL

執行了非法指令. 通常是因為可執行文件本身出現錯誤, 或者試圖執行數據段. 堆棧溢出時也有可能產生這個信號。

5) SIGTRAP

由斷點指令或其它trap指令產生. 由debugger使用。

6) SIGABRT

調用abort函數生成的信號。

7) SIGBUS

非法地址, 包括內存地址對齊(alignment)出錯。比如訪問一個四個字長的整數, 但其地址不是4的倍數。它與SIGSEGV的區別在於後者是由於對合法存儲地址的非法訪問觸發的(如訪問不屬於自己存儲空間或只讀存儲空間)。

8) SIGFPE

在發生致命的算術運算錯誤時發出. 不僅包括浮點運算錯誤, 還包括溢出及除數為0等其它所有的算術的錯誤。

9) SIGKILL

用來立即結束程序的運行. 本信號不能被阻塞、處理和忽略。如果管理員發現某個進程終止不了,可嘗試發送這個信號。

10) SIGUSR1

留給用戶使用

11) SIGSEGV

試圖訪問未分配給自己的內存, 或試圖往沒有寫權限的內存地址寫數據.

信號 11,即表示程序中可能存在特定條件下的非法內存訪問。

12) SIGUSR2

留給用戶使用

13) SIGPIPE

管道破裂。這個信號通常在進程間通信產生,比如採用FIFO(管道)通信的兩個進程,讀管道沒打開或者意外終止就往管道寫,寫進程會收到SIGPIPE信號。此外用Socket通信的兩個進程,寫進程在寫Socket的時候,讀進程已經終止。

14) SIGALRM

時鐘定時信號, 計算的是實際的時間或時鐘時間. alarm函數使用該信號.

15) SIGTERM

程序結束(terminate)信號, 與SIGKILL不同的是該信號可以被阻塞和處理。通常用來要求程序自己正常退出,shell命令kill預設產生這個信號。如果進程終止不了,我們才會嘗試SIGKILL。

17) SIGCHLD

子進程結束時, 父進程會收到這個信號。

如果父進程沒有處理這個信號,也沒有等待(wait)子進程,子進程雖然終止,但是還會在內核進程表中佔有表項,這時的子進程稱為殭屍 進程。這種情 況我們應該避免(父進程或者忽略SIGCHILD信號,或者捕捉它,或者wait它派生的子進程,或者父進程先終止,這時子進程的終止自動由init進程 來接管)。

18) SIGCONT

讓一個停止(stopped)的進程繼續執行. 本信號不能被阻塞. 可以用一個handler來讓程序在由stopped狀態變為繼續執行時完成特定的工作. 例如, 重新顯示提示符

19) SIGSTOP

停止(stopped)進程的執行. 注意它和terminate以及interrupt的區別:該進程還未結束, 只是暫停執行. 本信號不能被阻塞, 處理或忽略.

20) SIGTSTP

停止進程的運行, 但該信號可以被處理和忽略. 用戶鍵入SUSP字符時(通常是Ctrl-Z)發出這個信號

21) SIGTTIN

當後臺作業要從用戶終端讀數據時, 該作業中的所有進程會收到SIGTTIN信號. 預設時這些進程會停止執行.

22) SIGTTOU

類似於SIGTTIN, 但在寫終端(或修改終端模式)時收到.

23) SIGURG

有"緊急"數據或out-of-band數據到達socket時產生.

24) SIGXCPU

超過CPU時間資源限制. 這個限制可以由getrlimit/setrlimit來讀取/改變。

25) SIGXFSZ

當進程企圖擴大文件以至於超過文件大小資源限制。

26) SIGVTALRM

虛擬時鐘信號. 類似於SIGALRM, 但是計算的是該進程佔用的CPU時間.

27) SIGPROF

類似於SIGALRM/SIGVTALRM, 但包括該進程用的CPU時間以及系統調用的時間.

28) SIGWINCH

窗口大小改變時發出.

29) SIGIO

文件描述符準備就緒, 可以開始進行輸入/輸出操作.

30) SIGPWR

Power failure

31) SIGSYS

非法的系統調用。

三、 信號行為說明

不通的信號在不同的標準下,功能有所差別,下面列出主要的信號的默認行為和說明:

名稱數字標準默認行為說明SIGILL4ANSI終止+coredump執行了非法指令. 通常是因為可執行文件本身出現錯誤, 或者試圖執行數據段. 堆棧溢出時也有可能產生這個信號SIGABRT6ANSI終止+coredump調用abort函數生成的信號SIGBUS74.2 BSD終止+coredump非法地址, 包括內存地址對齊(alignment)出錯。比如訪問一個四個字長的整數, 但其地址不是4的倍數。它與SIGSEGV的區別在於後者是由於對合法存儲地址的非法訪問觸發的(如訪問不屬於自己存儲空間或只讀存儲空間)SIGFPE8ANSI終止+coredump在發生致命的算術運算錯誤時發出. 不僅包括浮點運算錯誤, 還包括溢出及除數為0等其它所有的算術的錯誤SIGSEGV11ANSI終止+coredump試圖訪問未分配給自己的內存, 或試圖往沒有寫權限的內存地址寫數據。訪問空指針,野指針基本都產生這個信號,也是最常見的信號SIGSTKFLT16N/A終止堆棧錯誤SIGPIPE13POSIX終止管道破裂。這個信號通常在進程間通信產生,比如採用FIFO(管道)通信的兩個進程,讀管道沒打開或者意外終止就往管道寫,寫進程會收到SIGPIPE信號。此外用Socket通信的兩個進程,寫進程在寫Socket的時候,讀進程已經終止SIGTRAP5POSIX終止+coredump由斷點指令或其它trap指令產生. 由debugger使用SIGHUP1POSIX終止用戶終端連接(正常或非正常)結束時發出, 通常是在終端的控制進程結束時, 通知同一session內的各個作業, 這時它們與控制終端不再關聯SIGINT2ANSI終止程序終止(interrupt)信號, 在用戶鍵入INTR字符(通常是Ctrl-C)時發出,用於通知前臺進程組終止進程SIGQUIT3POSIX終止+coredump和SIGINT類似, 但由QUIT字符(通常是Ctrl-)來控制. 進程在因收到SIGQUIT退出時會產生core文件, 在這個意義上類似於一個程序錯誤信號SIGKILL9POSIX終止用來立即結束程序的運行. 本信號不能被阻塞、捕獲和忽略。如果管理員發現某個進程終止不了,可嘗試發送這個信號SIGCHLD17POSIX忽略子進程結束時, 父進程會收到這個信號。如果父進程沒有處理這個信號,也沒有等待(wait)子進程,子進程雖然終止,但是還會在內核進程表中佔有表項,這時的子進程稱為殭屍進程。這種情 況我們應該避免(父進程或者忽略SIGCHILD信號,或者捕捉它,或者wait它派生的子進程,或者父進程先終止,這時子進程的終止自動由init進程來接管)SIGCONT18POSIX繼續/忽略讓一個停止(stopped)的進程繼續執行. 本信號不能被阻塞 . 可以用一個handler來讓程序在由stopped狀態變為繼續執行時完成特定的工作. 例如, 重新顯示提示符..在進程掛起時是繼續,否則是忽略SIGSTOP19POSIX暫停暫停進程的執行. 注意它和terminate以及interrupt的區別:該進程還未結束, 只是暫停執行. 本信號不能被阻塞、捕獲或忽略SIGALRM14POSIX終止時鐘定時信號, 計算的是實際的時間或時鐘時間. alarm函數使用該信號四、信號分類

在以上列出的信號中,程序不可捕獲、阻塞或忽略的信號有:

SIGKILL,SIGSTOP 

不能恢復至默認動作的信號有:

SIGILL,SIGTRAP 

默認會導致進程流產的信,有:

SIGABRT,SIGBUS,SIGFPE,SIGILL,SIGIOT,SIGQUIT,SIGSEGV,SIGTRAP,SIGXCPU,SIGXFSZ 

默認會導致進程退出的信號有:

SIGALRM,SIGHUP,SIGINT,SIGKILL,SIGPIPE,SIGPOLL,SIGPROF,SIGSYS,SIGTERM,SIGUSR1,SIGUSR2,SIGVTALRM 

默認會導致進程停止的信號有:

SIGSTOP,SIGTSTP,SIGTTIN,SIGTTOU 

默認進程忽略的信號有:

SIGCHLD,SIGPWR,SIGURG,SIGWINCH

此外,SIGIO在SVR4是退出,在4.3BSD中是忽略;

SIGCONT在進程掛起時是繼續,否則是忽略,不能被阻塞

終止程序的時候在不得已的情況下不能用SIGKILL,因為SIGKILL不會對子進程進行處理,只是把對自己進行處理。

五、信號驅動IO-SIGIO-29

下面我們主要講SIGIO-29的使用。

參考上圖:

時刻1 通過sigaction系統調用建立信號SIGIO的信號處理函數,該函數壺立即返回,注意,對應的驅動必須支持方法.fastnc時刻2 數據此時沒有準備好,應進程會繼續執行,而內核會繼續等待數據,也就是說等待數據階段應用進程是非阻塞的。時刻3 內核准備好了數據,要向應用進程複製數據,通過函數kill_fasync()向應用程式遞交SIGIO信號,二應用程式的信號處理程序會被調用到,在該函數中我們可以通過read等系統調用從內核賦值程序到進程時刻5 數據複製完成,會返回成功的指示,應用程式可以繼續處理數據

信號驅動 I/O 的 CPU 利用率很高,因為在圖中,等待數據的那段時間2,應用程式可以繼續執行其他操作。

六、程序實現 1. 信號註冊函數signal()
#include <signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);

功能:

給信號signum註冊處理函數,函數原型是void (*sighandler_t)(int)
當收到信號signum後,就會調用註冊的函數

參數:

int signum  信號值
sighandler_t handler  信號處理函數

2.內核函數
void kill_fasync(struct fasync_struct **fp, int sig, int band)

功能:

發送信號sig給進程,通知進程是可讀還是可寫,由band給出
POLLIN    :可讀
POLLOUT:可寫

通用字符設備的.fasync方法,一般都是固定的寫法,我們暫時可以不用關心他的原理,會用即可,具體寫法如下:

static ssize_t hello_write (struct file *filep, const char __user *buf, size_t size, loff_t *pos)
{
 int error;
…………
 kill_fasync(&hello_fasync,SIGIO,POLLIN);
 return size;
}

static struct file_operations hello_ops = 
{
…………
 .fasync = hello_fasync_func,
};

2. 源程序

驅動程序:hello.c

/*  
 *公眾號:一口Linux
 *2021.6.21
 *version: 1.0.0
*/
#include <linux/init.h>
#include <linux/module.h>
#include <linux/kdev_t.h>
#include <linux/fs.h>
#include <linux/cdev.h>
#include <linux/uaccess.h>
#include <linux/poll.h> 
#include<asm/signal.h>

static int major = 237;
static int minor = 0;
static dev_t devno;
static struct cdev cdev;
struct device *class_dev = NULL;
struct class *cls;

struct fasync_struct *hello_fasync;

static int hello_open (struct inode *inode, struct file *filep)
{
 printk("hello_open()\n");
 return 0;
}
static int hello_release (struct inode *inode, struct file *filep)
{
 printk("hello_release()\n");

 return 0;
}

#define KMAX_LEN 32
char kbuf[KMAX_LEN+1] = "kernel";


//read(fd,buff,40);

static ssize_t hello_read (struct file *filep, char __user *buf, size_t size, loff_t *pos)
{
 int error;

 
 if(size > strlen(kbuf))
 {
  size = strlen(kbuf);
 }

 if(copy_to_user(buf,kbuf, size))
 {
  error = -EFAULT;
  return error;
 }

 return size;
}
//write(fd,buff,40);
static ssize_t hello_write (struct file *filep, const char __user *buf, size_t size, loff_t *pos)
{
 int error;

 if(size > KMAX_LEN)
 {
  size = KMAX_LEN;
 }
 memset(kbuf,0,sizeof(kbuf));
 if(copy_from_user(kbuf, buf, size))
 {
  error = -EFAULT;
  return error;
 }
 printk("%s\n",kbuf);
 kill_fasync(&hello_fasync,SIGIO,POLLIN);
 return size;
}

int hello_fasync_func(int fd,struct file* filep,int on)
{
 printk("led_fasync \n");
 return fasync_helper(fd,filep,on,&hello_fasync);
}

static struct file_operations hello_ops = 
{
 .open = hello_open,
 .release = hello_release,
 .read = hello_read,
 .write = hello_write,
 .fasync = hello_fasync_func,
};
static int hello_init(void)
{
 int result;
 int error;
 
 printk("hello_init \n");
 result = register_chrdev( major, "hello", &hello_ops);
 if(result < 0)
 {
  printk("register_chrdev fail \n");
  return result;
 }
 cls = class_create(THIS_MODULE, "hellocls");
 if (IS_ERR(cls)) {
  printk(KERN_ERR "class_create() failed for cls\n");
  result = PTR_ERR(cls);
  goto out_err_1;
 }
 devno = MKDEV(major, minor);
 
 class_dev = device_create(cls, NULL, devno, NULL, "hellodev");
 if (IS_ERR(class_dev)) {
  result = PTR_ERR(class_dev);
  goto out_err_2;
 }
 
 return 0;

out_err_2:
 class_destroy(cls);
out_err_1:
 unregister_chrdev(major,"hello");
 return  result;
}
static void hello_exit(void)
{
 printk("hello_exit \n");
 device_destroy(cls, devno);
 class_destroy(cls);
 unregister_chrdev(major,"hello");
 return;
}
module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");
//proc/devices

write.c

/*  
 *一口Linux
 *2021.6.21
 *version: 1.0.0
*/

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
main()
{
 int fd;
 int len;
 char buf[64]={0};
 char buf2[64+1]="peng";
 
 
 fd = open("/dev/hellodev",O_RDWR);
 if(fd<0)
 {
  perror("open fail \n");
  return;
 }

 
 printf("before write\n");
 len = write(fd,buf2,strlen(buf2));
 printf("after write\n");

 printf("len = %d\n",len);
 
  
 close(fd);
}

test.c

/*  
 *公眾號:一口Linux
 *2021.6.21
 *version: 1.0.0
*/
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include<signal.h>

char buff[64] = {0};
int fd;

void func(int signo)
{
 printf("signo= %d\n",signo);
 read(fd,buff,sizeof(buff));
 printf("buff=%s\n",buff);
 return ;
}

main()
{
 int flage;

 fd = open("/dev/hellodev",O_RDWR);
 if(fd<0)
 {
  perror("open fail \n");
  return;
 }
 fcntl(fd,F_SETOWN,getpid());
 flage=fcntl(fd,F_GETFL);
 fcntl(fd,F_SETFL,flage|FASYNC);
    signal(SIGIO,func);
 while(1); 
 close(fd);
}

3. 執行結果

編譯

make
gcc test.c -o run
gcc write.c -o run

執行:

insmod hello.ko

先開啟一個終端 ,執行

./run

再開啟一個終端 ,執行

./w

執行結果如下:

可以看到,寫入數據後,信號處理程序被調用到,並且列印出信號的值29,同時從驅動力讀取出數據。

本例以字符設備為基礎來實現,詳細原理,請參考博主其他文章。

《Linux驅動系列文章合集》

完整代碼和執行環境,請關注,一口君的號,後臺回覆:ubuntu 

B站也有同步視頻。

相關焦點

  • 使用 GDB + Qemu 調試 Linux 內核
    Linux 內核編譯和文件系統製作2.1 Linux 內核編譯編譯內核和製作文件系統在 CentOS 7.7 的機器上。源碼從國內清華的源下載:http://ftp.sjtu.edu.cn/sites/ftp.kernel.org/pub/linux/kernel/, 此處選擇 linux-4.19.172.tar.gz 版本。
  • Linux 內核如何描述一個進程?
    它包含的欄位能完整地描述一個正在執行的程序,包括 cpu 狀態、打開的文件、地址空間、掛起的信號、進程狀態等。struct signal_struct *signal: 信號處理相關。其他欄位,等到有需要的時候再回過頭來學習。2.2 當發生系統調用或者進程切換時,內核如何找到 task_struct ?
  • Manjaro的Linux內核管理
    Manjaro不僅支持使用多個內核(可從開機畫面處選擇),也可以方便地安裝最新的內核。這需要藉助Manjaro的硬體檢測命令來完成。該命令的語法如下:sudo mhwd-kernel [-i] [新內核:linux(version)] [可選-刪除當前內核:rmc]當要用命令安裝的新內核時,不必寫完整的版本號。
  • Linux驅動05 | 內核編譯
    當前使用了一個發布版本的內核,如當前的內核為linux-3.4.39-gec,那麼這個時候開發板的內核也得是linux-3.4.39-gec,若版本不一致,內核模塊無法進行加載。6.out編譯成功後輸出鏡像的目錄7.prebuilts編譯uboot和內核使用到的交叉編譯工具8.prototype三星公司提供的一些底層源碼,用來支持編譯uboot和內核源碼9.tools一些工具,如mkimage(專門將zImage製作為uImage的工具,在內核測試的時候可以使用uImage
  • 編寫屬於你的第一個Linux內核模塊
    生成事件的代碼通過使用隊列將等待進程放回TASK_RUNNING狀態來喚醒它們。調度程序將在以後在某個地方選擇它們之一。Linux有多種非可運行狀態,最值得注意的是TASK_INTERRUPTIBLE(一個可以通過信號中斷的睡眠)和TASK_KILLABLE(一個可被殺死的睡眠中的進程)。所有這些都應該正確處理,並等待隊列為你做這些事。
  • 使用AFL對Linux內核Fuzzing的總結
    通常,在使用AFL時,我們需要檢測目標代碼,以便以AFL兼容的方式報告覆蓋範圍。但我們想要Fuzzing 內核!我們不能只用「afl-gcc」重新編譯它!。我們將準備一個二進位文件,讓AFL認為它是用它的工具編譯的。這個二進位文件將報告從內核中提取的代碼覆蓋率。
  • Linux 內核 / 進程管理 / 如何描述一個進程?
    它包含的欄位能完整地描述一個正在執行的程序,包括 cpu 狀態、打開的文件、地址空間、掛起的信號、進程狀態等。struct signal_struct *signal: 信號處理相關。其他欄位,等到有需要的時候再回過頭來學習。2.2 當發生系統調用或者進程切換時,內核如何找到 task_struct ?
  • linux 進程內核棧
    在執行系統調用陷入內核之後,這些內核代碼所使用的棧並不是原先用戶空間中的棧,而是一個內核空間的棧,這個稱作進程的「內核棧」。實際上在linux kernel中,task_struct、thread_info都用來保存進程相關信息,即進程PCB信息。然而不同的體系結構裡,進程需要存儲的信息不盡相同,linux使用task_struct存儲通用的信息,將體系結構相關的部分存儲在thread_info中。
  • 強網杯真題實例分析丨Linux內核提權技術
    2、結合強網杯真題進行實例分析;3、思考與總結。密碼復用另外需要注意的是很多管理員會重複使用密碼,所以有可能資料庫或者Web後臺的密碼就是root密碼。考慮到這些因素,如果管理員可以阻止將利用程序引入和/或執行到Linux文件系統上,則內核利用程序攻擊將不再可行。因此,管理員應專注於限制或刪除支持文件傳輸的程序,例如FTP,TFTP,SCP,wget和curl。當需要這些程序時,它們的使用應限於特定的用戶、目錄、應用程式(例如SCP)和特定的IP位址或域。
  • 如何在 Ubuntu/Linux Mint 中安裝最新 Linux內核
    它提供了一個帶有「主線mainline」內核更新列表的簡單界面,允許在基於 Ubuntu 的發行版中一鍵安裝、刪除或清除內核。如何在Ubuntu中安裝Mainline:注意:主線內核由Ubuntu內核團隊提供,用於測試和調試。它們不受支持,也不適合生產使用。您應該只在它們可能修復當前內核的一個嚴重問題時才安裝它們。
  • 實例講解丨PLC如何使用模擬量?
    再舉個例子,一些化工企業需要測量液體的酸鹼度,那麼酸鹼度不可以用數字量的0或者1表示,只能使用模擬量,PH6.5,PH7.0等等才可以清晰的描繪出實際的數據。那麼學習模擬量其實應該學習兩部分內容:一、硬體硬體包括選型和接線,選型就是針對不同的工況選擇不同的模擬量以及PLC,比如說模擬量使用0-10V,4-20ma還是0-20ma還是特殊的熱電偶信號等等。其次就是接線,大家不要小瞧了接線,針對不同的PLC有不同的接線方法。
  • Linux中的Lsmod命令(列出內核模塊)
    Linux內核具有模塊化設計。內核模塊(通常稱為驅動)是一段擴展內核功能的代碼。模塊要麼編譯為可加載模塊,要麼內置在內核中。可加載模塊可以根據需要在正在運行的內核中進行加載和卸載,而無需重新引導系統。通常,模塊由udev(設備管理器)按需加載。
  • 分享以原始patch開始深究Linux內核
    發現很多人都有一個困擾,對linux內核了解到一定程度,比如看過ULK,寫過幾年linux驅動,甚至在linux系統上工作過很多年的人,想對linux
  • 深入學習 Linux 內核模塊
    linux 中國 - LazyWolfLin,作者:Sandra Henry-Stockerhttps://linux.cn/article-11075-1.htmllsmod 命令能夠告訴你當前系統上加載了哪些內核模塊,以及關於使用它們的一些有趣的細節。
  • 《Linux設備驅動開發詳解:基於最新的Linux 4.0內核》china-pub上線預售
    由Linux以及圍繞著Linux進行產品研發的眾多企業和愛好者構成了一個龐大的Linux生態圈。而本書,無疑給這個龐大的生態圈注入了養料。  然而,養料的注入應該是持續不斷的。至今,Linux內核的底層BSP、驅動框架和內核實現發生了許多變更,本書涵蓋了這些新的變化,這將給予開發者更多新的幫助。內核的代碼不斷重構並最優化,而本書也無疑是一次重大的重構。
  • Linux 內存管理初探
    合理的使用內存,有助於提升機器的性能和穩定性。本文主要介紹 linux 內存組織結構和頁面布局,內存碎片產生原因和優化算法,linux 內核幾種內存管理的方法,內存使用場景以及內存使用的那些坑。從內存的原理和結構,到內存的算法優化,再到使用場景,去探尋內存管理的機制和奧秘。一、走進 linux 內存1、內存是什麼?
  • Linux內核Ramdisk(initrd)機制
    嵌入式linux QQ交流群:175159209,歡迎愛好者加入交流技術問題!
  • ubuntu降低你使用的內核到指定版本
    一般來說我們有幾種方式去安裝你對應的內核,從官網上下載.deb包安裝(下面有描述),也可以使用apt-cache search linux命令進行查找我們系統裡面是否有對應的安裝包選項,用apt-get install
  • 升級 Ubuntu Linux 內核的幾種不同方法 | Linux 中國
    然後下載符合以下格式的兩個文件(其中 X.Y.Z 是最高版本號):1. linux-image-X.Y.Z-generic-*.deb2. linux-modules-X.Y.Z-generic-.deb在終端中改變到文件所在的目錄,然後執行此命令手動安裝內核:$ sudo dpkg --install *.deb重啟系統,使用新內核:
  • 理解Linux內核搶佔模型(最透徹一篇)
    /#.XrKLcfnx05k作者:Liran B.H譯者:宋寶華當配置Linux內核的時候,我們可以選擇一些參數,這些參數能影響系統的行為。當你使用 make menuconfig配置內核的時候,你能看到這樣的菜單:為了深入理解這三個搶佔模型的區別,我們將寫一個案例:如果低優先級的線程陷入系統調用,高優先級的線程睡眠到期,究竟會發生什麼?下面我們來一種模型一種模型地看。