10.5. 中斷驅(qū)動(dòng) I/O

2018-02-24 15:50 更新

10.5.?中斷驅(qū)動(dòng) I/O

無論何時(shí)一個(gè)數(shù)據(jù)傳送到或自被管理的硬件可能因?yàn)槿魏卧蚨舆t, 驅(qū)動(dòng)編寫者應(yīng)當(dāng)實(shí)現(xiàn)緩存. 數(shù)據(jù)緩存幫助來分離數(shù)據(jù)傳送和接收從寫和讀系統(tǒng)調(diào)用, 并且整個(gè)系統(tǒng)性能受益.

一個(gè)好的緩存機(jī)制產(chǎn)生了中斷驅(qū)動(dòng)的 I/O, 一個(gè)輸入緩存在中斷時(shí)填充并且被讀取設(shè)備的進(jìn)程清空; 一個(gè)輸出緩存由寫設(shè)備的進(jìn)程填充并且在中斷時(shí)清空. 一個(gè)中斷驅(qū)動(dòng)的輸出的例子是 /dev/shortprint 的實(shí)現(xiàn).

為使中斷驅(qū)動(dòng)的數(shù)據(jù)傳送成功發(fā)生, 硬件應(yīng)當(dāng)能夠產(chǎn)生中斷, 使用下列語(yǔ)義:

  • 對(duì)于輸入, 設(shè)備中斷處理器, 當(dāng)新數(shù)據(jù)到達(dá)時(shí), 并且準(zhǔn)備好被系統(tǒng)處理器獲取. 進(jìn)行的實(shí)際動(dòng)作依賴是否設(shè)備使用 I/O 端口, 內(nèi)存映射, 或者 DMA.

  • 對(duì)于輸出, 設(shè)備遞交一個(gè)中斷, 或者當(dāng)它準(zhǔn)備好接受新數(shù)據(jù), 或者確認(rèn)一個(gè)成功的數(shù)據(jù)傳送. 內(nèi)存映射的和能DMA的設(shè)備常常產(chǎn)生中斷來告訴系統(tǒng)它們完成了這個(gè)緩存.

在一個(gè)讀或?qū)懪c實(shí)際數(shù)據(jù)到達(dá)之間的時(shí)間關(guān)系在第 6 章的"阻塞和非阻塞操作"一節(jié)中介紹.

10.5.1.?一個(gè)寫緩存例子

我們已經(jīng)幾次提及 shortprint 驅(qū)動(dòng); 現(xiàn)在是時(shí)候真正看看. 這個(gè)模塊為并口實(shí)現(xiàn)一個(gè)非常簡(jiǎn)單, 面向輸出的驅(qū)動(dòng); 它是足夠的, 但是, 來使能文件打印. 如果你選擇來測(cè)試這個(gè)驅(qū)動(dòng), 但是, 記住你必須傳遞給打印機(jī)一個(gè)文件以它理解的格式; 不是所有的打印機(jī)在給一個(gè)任意數(shù)據(jù)的流時(shí)很好響應(yīng).

shortprint 驅(qū)動(dòng)維護(hù)一個(gè)一頁(yè)的環(huán)形輸出緩存. 當(dāng)一個(gè)用戶空間進(jìn)程寫數(shù)據(jù)到這個(gè)設(shè)備, 數(shù)據(jù)被填入緩存, 但是寫方法實(shí)際沒有進(jìn)行任何 I/O. 相反, shortp_write 的核心看來如此:


while (written < count)
{
        /* Hang out until some buffer space is available. */
        space = shortp_out_space();
        if (space <= 0) {
                if (wait_event_interruptible(shortp_out_queue,
                                             (space = shortp_out_space()) > 0))
                        goto out;
        }

        /* Move data into the buffer. */
        if ((space + written) > count)
                space = count - written;

        if (copy_from_user((char *) shortp_out_head, buf, space)) {
                up(&shortp_out_sem);
                return -EFAULT;
        }
        shortp_incr_out_bp(&shortp_out_head, space);
        buf += space;
        written += space;

        /* If no output is active, make it active. */
        spin_lock_irqsave(&shortp_out_lock, flags);
        if (! shortp_output_active)
                shortp_start_output();
        spin_unlock_irqrestore(&shortp_out_lock, flags);
}

out:
*f_pos += written;

一個(gè)旗標(biāo) ( shortp_out_sem ) 控制對(duì)這個(gè)環(huán)形緩存的存取; shortp_write 就在上面的代碼片段之前獲得這個(gè)旗標(biāo). 當(dāng)持有這個(gè)旗標(biāo), 它試圖輸入數(shù)據(jù)到這個(gè)環(huán)形緩存. 函數(shù) shortp_out_space 返回可用的連續(xù)空間的數(shù)量(因此, 沒有必要擔(dān)心緩存回繞); 如果這個(gè)量是 0, 驅(qū)動(dòng)等到釋放一些空間. 它接著拷貝它能夠的數(shù)量的數(shù)據(jù)到緩存中.

一旦有數(shù)據(jù)輸出, shortp_write 必須確保數(shù)據(jù)被寫到設(shè)備. 數(shù)據(jù)的寫是通過一個(gè)工作隊(duì)列函數(shù)完成的; shortp_write 必須啟動(dòng)這個(gè)函數(shù)如果它還未在運(yùn)行. 在獲取了一個(gè)單獨(dú)的, 控制存取輸出緩存的消費(fèi)者一側(cè)(包括 shortp_output_active)的數(shù)據(jù)的自旋鎖后, 它調(diào)用 shortp_start_output 如果需要. 接著只是注意多少數(shù)據(jù)被寫到緩存并且返回.

啟動(dòng)輸出進(jìn)程的函數(shù)看來如下:


static void shortp_start_output(void)
{
        if (shortp_output_active) /* Should never happen */
                return;

        /* Set up our 'missed interrupt' timer */
        shortp_output_active = 1;
        shortp_timer.expires = jiffies + TIMEOUT;
        add_timer(&shortp_timer);

        /* And get the process going. */
        queue_work(shortp_workqueue, &shortp_work);
}

處理硬件的事實(shí)是, 你可以, 偶爾, 丟失來自設(shè)備的中斷. 當(dāng)發(fā)生這個(gè), 你確實(shí)不想你的驅(qū)動(dòng)一直停止直到系統(tǒng)重啟; 這不是一個(gè)用戶友好的做事方式. 最好是認(rèn)識(shí)到一個(gè)中斷已經(jīng)丟失, 收拾殘局, 繼續(xù). 為此, shortprint 甚至一個(gè)內(nèi)核定時(shí)器無論何時(shí)它輸出數(shù)據(jù)給設(shè)備. 如果時(shí)鐘超時(shí), 我們可能丟失一個(gè)中斷. 我們很快會(huì)看到定時(shí)器函數(shù), 但是, 暫時(shí), 讓我們堅(jiān)持在主輸出功能上. 那是在我們的工作隊(duì)列函數(shù)里實(shí)現(xiàn)的, 它, 如同你上面看到的, 在這里被調(diào)度. 那個(gè)函數(shù)的核心看來如下:


spin_lock_irqsave(&shortp_out_lock, flags);
/* Have we written everything? */
if (shortp_out_head == shortp_out_tail)
{ /* empty */
        shortp_output_active = 0;
        wake_up_interruptible(&shortp_empty_queue);
        del_timer(&shortp_timer);

}
/* Nope, write another byte */
else
        shortp_do_write();
/* If somebody's waiting, maybe wake them up. */
if (((PAGE_SIZE + shortp_out_tail -shortp_out_head) % PAGE_SIZE) > SP_MIN_SPACE)
{
        wake_up_interruptible(&shortp_out_queue);
}
spin_unlock_irqrestore(&shortp_out_lock, flags);

因?yàn)槲覀冊(cè)谑褂霉蚕碜兞康妮敵鲆粋?cè), 我們必須獲得自旋鎖. 接著我們看是否有更多的數(shù)據(jù)要發(fā)送; 如果無, 我們注意輸出不再激活, 刪除定時(shí)器, 并且喚醒任何在等待隊(duì)列全空的進(jìn)程(這種等待當(dāng)設(shè)備被關(guān)閉時(shí)結(jié)束). 如果, 相反, 有數(shù)據(jù)要寫, 我們調(diào)用 shortp_do_write 來實(shí)際發(fā)送一個(gè)字節(jié)到硬件.

接著, 因?yàn)槲覀兛赡茉谳敵鼍彺嬷杏锌臻e空間, 我們考慮喚醒任何等待增加更多數(shù)據(jù)給那個(gè)緩存的進(jìn)程. 但是我們不是無條件進(jìn)行喚醒; 相反, 我們等到有一個(gè)最低數(shù)量的空間. 每次我們從緩存拿出一個(gè)字節(jié)就喚醒一個(gè)寫者是無意義的; 喚醒進(jìn)程的代價(jià), 調(diào)度它運(yùn)行, 并且使它重回睡眠, 太高了. 相反, 我們應(yīng)當(dāng)?shù)鹊竭M(jìn)程能夠立刻移動(dòng)相當(dāng)數(shù)量的數(shù)據(jù)到緩存. 這個(gè)技術(shù)在緩存的, 中斷驅(qū)動(dòng)的驅(qū)動(dòng)中是普通的.

為完整起見, 這是實(shí)際寫數(shù)據(jù)到端口的代碼:


static void shortp_do_write(void)
{
        unsigned char cr = inb(shortp_base + SP_CONTROL);
        /* Something happened; reset the timer */
        mod_timer(&shortp_timer, jiffies + TIMEOUT);
        /* Strobe a byte out to the device */
        outb_p(*shortp_out_tail, shortp_base+SP_DATA);
        shortp_incr_out_bp(&shortp_out_tail, 1);
        if (shortp_delay)
                udelay(shortp_delay);
        outb_p(cr | SP_CR_STROBE, shortp_base+SP_CONTROL);
        if (shortp_delay)
                udelay(shortp_delay);
        outb_p(cr & ~SP_CR_STROBE, shortp_base+SP_CONTROL);
}

這里, 我們復(fù)位定時(shí)器來反映一個(gè)事實(shí), 我們已經(jīng)作了一些處理, 輸送字節(jié)到設(shè)備, 并且更新了環(huán)形緩存指針.

工作隊(duì)列函數(shù)沒有直接重新提交它自己, 因此只有一個(gè)單個(gè)字節(jié)會(huì)被寫入設(shè)備. 在某一處, 打印機(jī)將, 以它的緩慢方式, 消耗這個(gè)字節(jié)并且準(zhǔn)備好下一個(gè); 它將接著中斷處理器. shortprint 中使用的中斷處理是簡(jiǎn)短的:


static irqreturn_t shortp_interrupt(int irq, void *dev_id, struct pt_regs *regs)
{
        if (! shortp_output_active)
                return IRQ_NONE;
        /* Remember the time, and farm off the rest to the workqueue function */
        do_gettimeofday(&shortp_tv);
        queue_work(shortp_workqueue, &shortp_work);
        return IRQ_HANDLED;
}

因?yàn)椴⒖诓灰笠粋€(gè)明顯的中斷確認(rèn), 中斷處理所有真正需要做的是告知內(nèi)核來再次運(yùn)行工作隊(duì)列函數(shù).

如果中斷永遠(yuǎn)不來如何? 至此我們已見到的驅(qū)動(dòng)代碼將簡(jiǎn)單地停止. 為避免發(fā)生這個(gè), 我們?cè)O(shè)置了一個(gè)定時(shí)器在幾頁(yè)前. 當(dāng)定時(shí)器超時(shí)運(yùn)行的函數(shù)是:


static void shortp_timeout(unsigned long unused) 
{
        unsigned long flags;
        unsigned char status;
        if (! shortp_output_active)
                return;
        spin_lock_irqsave(&shortp_out_lock, flags);
        status = inb(shortp_base + SP_STATUS);

        /* If the printer is still busy we just reset the timer */
        if ((status & SP_SR_BUSY) == 0 || (status & SP_SR_ACK)) {

                shortp_timer.expires = jiffies + TIMEOUT;
                add_timer(&shortp_timer);
                spin_unlock_irqrestore(&shortp_out_lock, flags);
                return;
        }
        /* Otherwise we must have dropped an interrupt. */
        spin_unlock_irqrestore(&shortp_out_lock, flags);
        shortp_interrupt(shortp_irq, NULL, NULL);
}

如果沒有輸出要被激活, 定時(shí)器函數(shù)簡(jiǎn)單地返回. 這避免了定時(shí)器重新提交自己, 當(dāng)事情在被關(guān)閉時(shí). 接著, 在獲得了鎖之后, 我們查詢端口的狀態(tài); 如果它聲稱忙, 它完全還沒有時(shí)間來中斷我們, 因此我們復(fù)位定時(shí)器并且返回. 打印機(jī)能夠, 有時(shí), 花很長(zhǎng)時(shí)間來使自己準(zhǔn)備; 考慮一下缺紙的打印機(jī), 而每個(gè)人在一個(gè)長(zhǎng)周末都不在. 在這種情況下, 只有耐心等待直到事情改變.

但是, 如果打印機(jī)聲稱準(zhǔn)備好了, 我們一定丟失了它的中斷. 這個(gè)情況下, 我們簡(jiǎn)單地手動(dòng)調(diào)用我們的中斷處理來使輸出處理再動(dòng)起來.

shortpirnt 驅(qū)動(dòng)不支持從端口讀數(shù)據(jù); 相反, 它象 shortint 并且返回中斷時(shí)間信息. 但是一個(gè)中斷驅(qū)動(dòng)的讀方法的實(shí)現(xiàn)可能非常類似我們已經(jīng)見到的. 從設(shè)備來的數(shù)據(jù)可能被讀入驅(qū)動(dòng)緩存; 它可能被拷貝到用戶空間只在緩存中已經(jīng)累積了相當(dāng)數(shù)量的數(shù)據(jù), 完整的讀請(qǐng)求已被滿足, 或者某種超時(shí)發(fā)生.

以上內(nèi)容是否對(duì)您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)