日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > linux >内容正文

linux

基于 Linux 和 MiniGUI 的嵌入式系统软件开发指南(六) MiniGUI 提供的非 GUI/GDI 接口...

發布時間:2025/3/21 linux 78 豆豆
生活随笔 收集整理的這篇文章主要介紹了 基于 Linux 和 MiniGUI 的嵌入式系统软件开发指南(六) MiniGUI 提供的非 GUI/GDI 接口... 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

1 引言
一般而言,GUI 系統的應用程序編程接口主要集中于窗口、消息隊列、圖形設備等相關方面。但因為 GUI 系統在處理系統事件時通常會提供自己的機制,而這些機制往往會和操作系統本身提供的機制不相兼容。比如,MiniGUI 提供了消息循環機制,而應用程序的結構一般是消息驅動的;也就是說,應用程序通過被動接收消息來工作。但很多情況下,應用程序需要主動監視某個系統事件, 比如在 UNIX 操作系統中,可以通過 select 系統調用監聽某個文件描述符上是否有可讀數據。這樣,如何將 MiniGUI 的消息隊列機制和現有操作系統的其他機制融合在一起,就成了一個較為困難的問題。本文將講述幾種解決這一問題的方法。

我們知道,MiniGUI-Lite 采用 UNIX Domain Socket 實現客戶程序和服務器程序之間的交互。應用程序也可以利用這一機制,完成自己的通訊任務――客戶向服務器提交請求,而服務器完成對客戶的請求處理并應答。 一方面,在 MiniGUI-Lite 的服務器程序中,你可以擴展這一機制,注冊自己的請求處理函數,完成定制的請求/響應通訊任務。另一方面,MiniGUI-Lite 當中也提供了若干用來創建和操作 UNIX Domain Socket 的函數,任何 MiniGUI-Lite 的應用程序都可以建立 UNIX Domain Socket,并完成和其他 MiniGUI-Lite 應用程序之間的數據交換。本文將舉例講述如何利用 MiniGUI-Lite 提供的函數完成此類通訊任務。

嵌入式 Linux 系統現在能夠在許多不同架構的硬件平臺上運行,MiniGUI 也能夠在這些硬件平臺上運行。但由于許多硬件平臺具有和其他硬件平臺不同的特性,比如說,常見的 CPU 是 Little Endian 的,而某些 CPU 則是 Big Endian 的。這要求我們在編寫代碼,尤其是文件 I/O 相關代碼時,必須編寫可移植代碼,以便適合具有不同架構的平臺。本文將描述 MiniGUI 為應用程序提供的可移植性函數及其用法。

除了與上述內容相關的函數之外,MiniGUI 還提供了其他一些函數,本文最后部分將描述這些函數的用途和用法,包括配置文件讀寫以及定點數運算。

2 MiniGUI-Lite和 select 系統調用
我們知道,在 MiniGUI-Lite 之上運行的應用程序只有一個消息隊列。應用程序在初始化之后,會建立一個消息循環,然后不停地從這個消息隊列當中獲得消息并處理,直到接收到 MSG_QUIT 消息為止。應用程序的窗口過程在處理消息時,要在處理完消息之后立即返回,以便有機會獲得其他的消息并處理。現在,如果應用程序在處理某個消息時監聽某個 文件描述符而調用 select 系統調用,就有可能會出現問題――因為 select 系統調用可能會長時間阻塞,而由 MiniGUI-Lite 服務器發送給客戶的事件得不到及時處理。這樣,消息驅動的方式和 select 系統調用就難于很好地融合。在 MiniGUI-Threads 中,因為每個線程都有自己相應的消息隊列,而系統消息隊列是由單獨運行的 desktop 線程管理的,所以任何一個應用程序建立的線程都可以長時間阻塞,從而可以調用類似 select 的系統調用。但在 MiniGUI-Lite 當中,如果要監聽某個應用程序自己的文件描述符事件,必須進行恰當的處理,以避免長時間阻塞。

在 MiniGUI-Lite 當中,有幾種解決這一問題的辦法:

  • 在調用 select 系統調用時,傳遞超時值,保證 select 系統調用不會長時間阻塞。
  • 設置定時器,定時器到期時,利用 select 系統調用查看被監聽的文件描述符。如果沒有相應的事件發生,則立即返回,否則進行讀寫操作。
  • 利用 MiniGUI-Lite 提供的 RegisterListenFD 函數在系統中注冊監聽文件描述符,并在被監聽的文件描述符上發生指定的事件時,向某個窗口發送 MSG_FDEVENT 消息。
  • 由于前兩種解決方法比較簡單,這里我們重點講述的第三種解決辦法。MiniGUI-Lite 為應用程序提供了如下兩個函數及一個宏:


    #define MAX_NR_LISTEN_FD 5

    /* Return TRUE if all OK, and FALSE on error. */
    BOOL GUIAPI RegisterListenFD (int fd, int type, HWND hwnd, void* context);

    /* Return TRUE if all OK, and FALSE on error. */
    BOOL GUIAPI UnregisterListenFD (int fd);
    • MAX_NR_LISTEN_FD 宏定義了系統能夠監聽的最多文件描述符數,默認定義為 5。
    • RegisterListenFD 函數在系統當中注冊一個需要監聽的文件描述符,并指定監聽的事件類型(type 參數,可取 POLLIN、POLLOUT 或者 POLLERR),接收 MSG_FDEVENT 消息的窗口句柄以及一個上下文信息。
    • UnregisterListenFD 函數注銷一個被注冊的監聽文件描述符。

    在應用程序使用RegisterListenFD 函數注冊了被監聽的文件描述符之后,當指定的事件發生在該文件描述符上時,系統會將 MSG_FDEVENT 消息發送到指定的窗口,應用程序可在窗口過程中接收該消息并處理。MiniGUI 中的 libvcongui 就利用了上述函數監聽來自主控偽終端上的可讀事件,如下面的程序段所示(vcongui/vcongui.c):


    ...

    /* 注冊主控偽終端偽監聽文件描述符 */
    RegisterListenFD (pConInfo->masterPty, POLLIN, hMainWnd, 0);

    /* 進入消息循環 */
    while (!pConInfo->terminate && GetMessage (&Msg, hMainWnd)) {
    DispatchMessage (&Msg);
    }
    /* 注銷監聽文件描述符 */
    UnregisterListenFD (pConInfo->masterPty);

    ...

    /* 虛擬控制臺的窗口過程 */
    static int VCOnGUIMainWinProc (HWND hWnd, int message, WPARAM wParam, LPARAM lParam)
    {
    PCONINFO pConInfo;

    pConInfo = (PCONINFO)GetWindowAdditionalData (hWnd);
    switch (message) {

    ...

    /* 接收到 MSG_FDEVENT 消息,則處理主控偽終端上的輸入數據 */
    case MSG_FDEVENT:
    ReadMasterPty (pConInfo);
    break;

    ...
    }

    /* 調用默認窗口過程 */
    if (pConInfo->DefWinProc)
    return (*pConInfo->DefWinProc)(hWnd, message, wParam, lParam);
    else
    return DefaultMainWinProc (hWnd, message, wParam, lParam);
    }
    在 3.2 節當中,我們還可以看到RegisterListenFD 函數的使用。顯然,通過這種簡單的注冊監聽文件描述符的接口,MiniGUI-Lite 程序能夠方便地利用底層的消息機制完成對異步事件的處理。

    3 MiniGUI-Lite 與進程間通訊

    3.1 簡單請求/應答處理
    我們知道,MiniGUI-Lite 利用了 UNIX Domain Socket 實現服務器和客戶程序之間的通訊。為了實現客戶和服務器之間的簡單方便的通訊,MiniGUI-Lite 中定義了一種簡單的請求/響應結構。客戶程序通過指定的結構將請求發送到服務器,服務器處理請求并應答。在客戶端,一個請求定義如下 (include/gdi.h):


    typedef struct tagREQUEST {
    int id;
    const void* data;
    size_t len_data;
    } REQUEST;
    typedef REQUEST* PREQUEST;

    其中,id 是用來標識請求類型的整型數,data 是發送給該請求的關聯數據,len_data 則是數據的長度。客戶在初始化 REQUEST 結構之后,就可以調用 cli_request 向服務器發送請求,并等待服務器的應答。該函數的原型如下。


    /* send a request to server and wait reply */
    int cli_request (PREQUEST request, void* result, int len_rslt);

    服務器程序(即 mginit)會在自己的消息循環當中獲得來自客戶的請求,并進行處理,最終會將處理結果發送給客戶。

    在上述這種簡單的客戶/服務器通訊中,客戶和服務器必須就每個請求類型達成一致,也就是說,客戶和服務器必須了解每種類型請求的數據含義并進行恰當的處理。

    MiniGUI-Lite 利用上述這種簡單的通訊方法,實現了若干系統級的通訊任務:

    • 鼠標光標的管理。鼠標光標是一個全局資源,當客戶需要創建或者銷毀鼠標光標,改變鼠標光標的形狀、位置,顯示或者隱藏鼠標時,就發送請求到服務器,服務器程序完成相應任務并將結果發送給客戶。
    • 層及活動客戶管理。當客戶查詢層的信息,新建層,加入某個已有層,或者設置層中的活動客戶時,通過該接口發送請求到服務器。
    • 其他一些系統級的任務。比如在新的 GDI 接口中,服務器程序統一管理顯示卡中可能用來建立內存 DC 的顯示內存,當客戶要申請建立在顯示內存中的內存 DC 時,就會發送請求到服務器。

    為了讓應用程序也能夠通過這種簡單的方式實現客戶和服務器之間的通訊,服務器程序可以注冊一些定制的請求處理函數,然后客戶就可以向服務器發送這些請求。為此,MiniGUI-Lite 提供了如下接口:


    #define MAX_SYS_REQID 0x0010
    #define MAX_REQID 0x0018

    /*
    * Register user defined request handlers for server
    * Note that user defined request id should larger than MAX_SYS_REQID
    */
    typedef int (* REQ_HANDLER) (int cli, int clifd, void* buff, size_t len);
    BOOL GUIAPI RegisterRequestHandler (int req_id, REQ_HANDLER your_handler);
    REQ_HANDLER GUIAPI GetRequestHandler (int req_id);

    服務器可以通過調用RegisterRequestHandler 函數注冊一些請求處理函數。注意請求處理函數的原型由REQ_HANDLER 定義。還要注意系統定義了MAX_SYS_REQID 和 MAX_REQID 這兩個宏。MAX_REQID 是能夠注冊的最大請求 ID 號,而 MAX_SYS_REQID 是系統內部使用的最大的請求 ID 號,也就是說,通過RegisterRequestHandler 注冊的請求 ID 號,必須大于 MAX_SYS_REQID 而小于或等于 MAX_REQID。

    作為示例,我們假設服務器替客戶計算兩個整數的和。客戶發送兩個整數給服務器,而服務器將兩個整數的和發送給客戶。下面的程序段在服務器程序中運行,在系統中注冊了一個請求處理函數:


    typedef struct TEST_REQ
    {
    int a, b;
    } TEST_REQ;

    static int send_reply (int clifd, void* reply, int len)
    {
    MSG reply_msg = {HWND_INVALID, 0};

    /* 發送一個空消息接口給客戶,以便說明這是一個請求的應答 */
    if (sock_write (clifd, &reply_msg, sizeof (MSG)) < 0)
    return SOCKERR_IO;

    /* 將結果發送給客戶 */
    if (sock_write (clifd, reply, len) < 0)
    return SOCKERR_IO;

    return SOCKERR_OK;
    }

    static int test_request (int cli, int clifd, void* buff, size_t len)
    {
    int ret_value = 0;
    TEST_REQ* test_req = (TEST_REQ*)buff;

    ret_value = test_req.a + test_req.b;

    return send_reply (clifd, &ret_value, sizeof (int));
    }

    ...
    RegisterRequestHandler (MAX_SYS_REQID + 1, test_request);
    ...

    而客戶程序可以通過如下的程序段向客戶發送一個請求獲得兩個整數的和:


    REQUEST req;
    TEST_REQ test_req = {5, 10};
    int ret_value;

    req.id = MAX_SYS_REQID + 1;
    req.data = &rest_req;
    req.len_data = sizeof (TEST_REQ);

    cli_request (&req, &ret_value, sizeof (int));
    printf ("the returned value: %d\n", ret_value); /* ret_value 的值應該是 15 */

    讀者已經看到,通過這種簡單的請求/應答技術,MiniGUI-Lite 客戶程序和服務器程序之間可以建立一種非常方便的進程間通訊機制。但這種技術也有一些缺點,比如受到 MAX_REQID 大小的影響,通訊機制并不是非常靈活,而且請求只能發送給MiniGUI-Lite 的服務器程序(即 mginit)處理等等。

    3.2 復雜的 UNIX Domain Socket 封裝
    為了解決上述簡單請求/應答機制的不足,MiniGUI-Lite 也提供了經過封裝的 UNIX Domain Socket 處理函數。這些函數的接口原型如下(include/minigui.h):


    /* Used by server to create a listen socket.
    * Name is the name of listen socket.
    * Please located the socket in /var/tmp directory. */

    /* Returns fd if all OK, -1 on error. */
    int serv_listen (const char* name);

    /* Wait for a client connection to arrive, and accept it.
    * We also obtain the client's pid and user ID from the pathname
    * that it must bind before calling us. */

    /* returns new fd if all OK, < 0 on error */
    int serv_accept (int listenfd, pid_t *pidptr, uid_t *uidptr);

    /* Used by clients to connect to a server.
    * Name is the name of the listen socket.
    * The created socket will located at the directory /var/tmp,
    * and with name of '/var/tmp/xxxxx-c', where 'xxxxx' is the pid of client.
    * and 'c' is a character to distinguish diferent projects.
    * MiniGUI use 'a' as the project character.
    */

    /* Returns fd if all OK, -1 on error. */
    int cli_conn (const char* name, char project);

    #define SOCKERR_IO -1
    #define SOCKERR_CLOSED -2
    #define SOCKERR_INVARG -3
    #define SOCKERR_OK 0

    /* UNIX domain socket I/O functions. */

    /* Returns SOCKERR_OK if all OK, < 0 on error.*/
    int sock_write_t (int fd, const void* buff, int count, unsigned int timeout);
    int sock_read_t (int fd, void* buff, int count, unsigned int timeout);

    #define sock_write(fd, buff, count) sock_write_t(fd, buff, count, 0)
    #define sock_read(fd, buff, count) sock_read_t(fd, buff, count, 0)

    上述函數是 MiniGUI-Lite 用來建立系統內部使用的 UNIX Domain Socket 并進行數據傳遞的函數,是對基本套接字系統調用的封裝。這些函數的功能描述如下:

    • serv_listen:服務器調用該函數建立一個監聽套接字,并返回套接字文件描述符。建議將服務器監聽套接字建立在 /var/tmp/ 目錄下。
    • serv_accept:服務器調用該函數接受來自客戶的連接請求。
    • cli_conn:客戶調用該函數連接到服務器,其中 name 是客戶的監聽套接字。該函數為客戶建立的套接字將保存在 /var/tmp/ 目錄中,并且以 -c 的方式命名,其中 c 是用來區別不同套接字通訊用途的字母,由 project 參數指定。MiniGUI-Lite 內部使用了 'a',所以由應用程序建立的套接字,應該使用除 'a' 之外的字母。
    • sock_write_t:在建立并連接之后,客戶和服務器之間就可以使用 sock_write_t 函數和 sock_read_t 函數進行數據交換。sock_write_t 的參數和系統調用 write 類似,但可以傳遞進入一個超時參數,注意該參數以 10ms 為單位,為零時超時設置失效,且超時設置只在 mginit 程序中有效。
    • sock_read_t:sock_read_t 的參數和系統調用 read類似,但可以傳遞進入一個超時參數,注意該參數以 10ms 為單位,為零時超時設置失效,且超時設置只在 mginit 程序中有效。

    下面的代碼演示了作為服務器的程序如何利用上述函數建立一個監聽套接字:


    #define LISTEN_SOCKET "/var/tmp/mysocket"

    static int listen_fd;

    BOOL listen_socket (HWND hwnd)
    {
    if ((listen_fd = serv_listen (LISTEN_SOCKET)) < 0)
    return FALSE;
    return RegisterListenFD (fd, POLL_IN, hwnd, NULL);
    }

    當服務器接收到來自客戶的連接請求是,服務器的 hwnd 窗口將接收到 MSG_FDEVENT 消息,這時,服務器可接受該連接請求:


    int MyWndProc (HWND hwnd, int message, WPARAM wParam, LPARAM lParam)
    {
    switch (message) {

    ...

    case MSG_FDEVENT:
    if (LOWORD (wParam) == listen_fd) { /* 來自監聽套接字 */
    pid_t pid;
    uid_t uid;
    int conn_fd;
    conn_fd = serv_accept (listen_fd, &pid, &uid);
    if (conn_fd >= 0) {
    RegisterListenFD (conn_fd, POLL_IN, hwnd, NULL);
    }
    }
    else { /* 來自已連接套接字 */
    int fd = LOWORD(wParam);
    /* 處理來自客戶的數據 */
    sock_read_t (fd, ...);
    sock_write_t (fd, ....);
    }
    break;

    ...

    }
    }

    上面的代碼中,服務器將連接得到的新文件描述符也注冊為監聽描述符,因此,在 MSG_FDEVENT 消息的處理中,應該判斷導致 MSG_FDEVENT 消息的文件描述符類型,并做適當的處理。

    在客戶端,當需要連接到服務器時,可通過如下代碼:


    int conn_fd;

    if ((conn_fd = cli_conn (LISTEN_SOCKET, 'b')) >= 0) {
    /* 向服務器發送請求 */
    sock_write_t (fd, ....);
    /* 獲取來自服務器的處理結果 */
    sock_read_t (fd, ....);
    }

    4 編寫可移植代碼
    我們知道,許多嵌入式系統所使用的 CPU 具有和普通臺式機 CPU 完全不同的構造和特點。但有了操作系統和高級語言,可以最大程度上將這些不同隱藏起來。只要利用高級語言編程,編譯器和操作系統能夠幫助程序員解決許多和 CPU 構造及特點相關的問題,從而節省程序開發時間,并提高程序開發效率。然而某些 CPU 特點卻是應用程序開發人員所必須面對的,這其中就有如下幾個需要特別注意的方面:

    • 字節順序。一般情況下,我們接觸到的 CPU 在存放多字節的整數數據時,將低位字節存放在低地址單元中,比如常見的 Intel x86 系列 CPU。而某些 CPU 采用相反的字節順序。比如在嵌入式系統中使用較為廣泛的 PowerPC 就將低位字節存放在高地址單元中。前者叫 Little Endian 系統;而后者叫 Big Endian 系統。
    • 在某些平臺上的 Linux 內核,可能缺少某些高級系統調用,最常見的就是與虛擬內存機制相關的系統調用。在某些 CPU 上運行的 Linux 操作系統,因為 CPU 能力的限制,無法提供虛擬內存機制,基于虛擬內存實現的某些 IPC 機制就無法正常工作。比如在某些缺少 MMU 單元的 CPU 上,就無法提供 System V IPC 機制中的共享內存。

    為了編寫具有最廣泛適應性的可移植代碼,應用程序開發人員必須注意到這些不同,并且根據情況編寫可移植代碼。這里,我們將描述如何在 MiniGUI 應用程序中編寫可移植代碼。

    4.1 理解并使用 MiniGUI 的 Endian 讀寫函數
    為了解決上述的第一個問題,MiniGUI 提供了若干 Endian 相關的讀寫函數。這些函數可以劃分為如下兩類:

    • 用來交換字節序的函數。包括ArchSwapLE16、ArchSwapBE16 等。
    • 用來讀寫標準I/O 流的函數。包括MGUI_ReadLE16、MGUI_ReadBE16 等。

    前一類用來將某個 16位、32 位或者 64 位整數從某個特定的字節序轉換為系統私有(native)字節序。舉例如下:


    int fd, len_header;

    ...

    if (read (fd, &len_header, sizeof (int)) == -1)
    goto error;
    #if MGUI_BYTEORDER == MGUI_BIG_ENDIAN
    len_header = ArchSwap32 (len_header); // 如果是 Big Endian 系統,則轉換字節序
    #endif
    ...

    在上面的程序段中,首先通過 read 系統調用從指定的文件描述符中讀取一個整數值到 len_header 變量中。該文件中保存的整數值是 Little Endian 的,因此如果在 Big Endian 系統上使用這個整數值,就必須進行字節順序交換。這里可以使用 ArchSwapLE32,將 Little Endian 的 32 位整數值轉換為系統私有的字節序。也可以如上述程序段那樣,只對 Big Endian 系統進行字節序轉換,這時,只要利用 ArchSwap32 函數即可。

    MiniGUI 提供的用來轉換字節序的函數(或者宏)如下:

    • ArchSwapLE16(X) 將指定的以 Little Endian 字節序存放的 16 位整數值轉換為系統私有整數值。如果系統本身是 Little Endian 系統,則該函數不作任何工作,直接返回 X;如果系統本身是 Big Endian 系統,則調用 ArchSwap16 函數交換字節序。
    • ArchSwapLE32(X) 將指定的以 Little Endian 字節序存放的 32 位整數值轉換為系統私有整數值。如果系統本身是 Little Endian 系統,則該函數不作任何工作,直接返回 X;如果系統本身是 Big Endian 系統,則調用 ArchSwap32 函數交換字節序。
    • ArchSwapBE16(X) 將指定的以 Big Endian 字節序存放的 16 位整數值轉換為系統私有整數值。如果系統本身是 Big Endian 系統,則該函數不作任何工作,直接返回 X;如果系統本身是 Little Endian 系統,則調用 ArchSwap16 函數交換字節序。
    • ArchSwapBE32(X) 將指定的以 Big Endian 字節序存放的 32 位整數值轉換為系統私有整數值。如果系統本身是 Big Endian 系統,則該函數不作任何工作,直接返回 X;如果系統本身是 Little Endian 系統,則調用 ArchSwap32 函數交換字節序。

    MiniGUI 提供的第二類函數用來從標準 I/O 的文件對象中讀寫 Endian 整數值。如果要讀取的文件是以 Little Endian 字節序存放的,則可以使用 MGUI_ReadLE16 和MGUI_ReadLE32 等函數讀取整數值,這些函數將把讀入的整數值轉換為系統私有字節序,反之使用MGUI_ReadBE16 和MGUI_ReadBE32 函數。如果要寫入的文件是以 Little Endian 字節序存放的,則可以使用 MGUI_WriteLE16 和MGUI_WriteLE32 等函數讀取整數值,這些函數將把要寫入的整數值從系統私有字節序轉換為 Little Endian 字節序,然后寫入文件,反之使用MGUI_WriteBE16 和MGUI_WriteBE32 函數。下面的代碼段說明了上述函數的用法:


    FILE* out;
    int ount;
    ...
    MGUI_WriteLE32 (out, count); // 以 Little Endian 字節序保存 count 到文件中。
    ...

    4.2 利用條件編譯編寫可移植代碼
    在涉及到可移植性問題的時候,有時我們能夠方便地通過 4.1 中描述的方法進行函數封裝,從而提供具有良好移植性的代碼,但有時我們無法通過函數封裝的方法提供可移植性代碼。這時,恐怕只能使用條件編譯了。下面的代 碼說明了如何使用條件編譯的方法確保程序正常工作(該代碼來自 MiniGUI src/kernel/sharedres.c):


    /* 如果系統不支持共享內存,則定義 _USE_MMAP
    #undef _USE_MMAP
    /* #define _USE_MMAP 1 */

    void *LoadSharedResource (void)
    {
    #ifndef _USE_MMAP
    key_t shm_key;
    void *memptr;
    int shmid;
    #endif

    /* 裝載共享資源 */
    ...

    #ifndef _USE_MMAP /* 獲取共享內存對象 */
    if ((shm_key = get_shm_key ()) == -1) {
    goto error;
    }
    shmid = shmget (shm_key, mgSizeRes, SHM_PARAM | IPC_CREAT | IPC_EXCL);
    if (shmid == -1) {
    goto error;
    }

    // Attach to the share memory.
    memptr = shmat (shmid, 0, 0);
    if (memptr == (char*)-1)
    goto error;
    else {
    memcpy (memptr, mgSharedRes, mgSizeRes);
    free (mgSharedRes);
    }

    if (shmctl (shmid, IPC_RMID, NULL) < 0)
    goto error;
    #endif

    /* 打開文件 */
    if ((lockfd = open (LOCKFILE, O_WRONLY | O_CREAT | O_TRUNC, 0644)) == -1)
    goto error;

    #ifdef _USE_MMAP
    /* 如果使用 mmap,就將共享資源寫入文件 */
    if (write (lockfd, mgSharedRes, mgSizeRes) < mgSizeRes)
    goto error;
    else
    {
    free(mgSharedRes);
    mgSharedRes = mmap( 0, mgSizeRes, PROT_READ|PROT_WRITE, MAP_SHARED, lockfd, 0);
    }
    #else
    /* 否則將共享內存對象 ID 寫入文件 */
    if (write (lockfd, &shmid, sizeof (shmid)) < sizeof (shmid))
    goto error;
    #endif

    close (lockfd);

    #ifndef _USE_MMAP
    mgSharedRes = memptr;
    SHAREDRES_SHMID = shmid;
    #endif
    SHAREDRES_SEMID = semid;

    return mgSharedRes;

    error:
    perror ("LoadSharedResource");
    return NULL;
    }

    上述程序段是 MiniGUI-Lite 服務器程序用來裝載共享資源的。如果系統支持共享內存,則初始化共享內存對象,并將裝載的共享資源關聯到共享內存對象,然后將共享內存對象 ID 寫入文件;如果系統不支持共享內存,則將初始化后的共享資源全部寫入文件。在客戶端,如果支持共享內存,則可以從文件中獲得共享內存對象 ID,并直接關聯到共享內存;如果不支持共享內存,則可以使用 mmap 系統調用,將文件映射到進程的地址空間。客戶端的代碼段如下:


    void* AttachSharedResource (void)
    {
    #ifndef _USE_MMAP
    int shmid;
    #endif
    int lockfd;
    void* memptr;

    if ((lockfd = open (LOCKFILE, O_RDONLY)) == -1)
    goto error;

    #ifdef _USE_MMAP
    /* 使用 mmap 將共享資源映射到進程地址空間 */
    mgSizeRes = lseek (lockfd, 0, SEEK_END );
    memptr = mmap( 0, mgSizeRes, PROT_READ, MAP_SHARED, lockfd, 0);
    #else
    /* 否則獲取共享內存對象 ID,并關聯該共享內存 */
    if (read (lockfd, &shmid, sizeof (shmid)) < sizeof (shmid))
    goto error;
    close (lockfd);

    memptr = shmat (shmid, 0, SHM_RDONLY);
    #endif
    if (memptr == (char*)-1)
    goto error;
    return memptr;

    error:
    perror ("AttachSharedResource");
    return NULL;
    }

    5 其他

    5.1 讀寫配置文件
    MiniGUI 的配置文件,即 /etc/MiniGUI.cfg 文件的格式,采用了類似 Windows INI 文件的格式。這種文件格式非常簡單,如下所示:


    [section-name1]
    key-name1=key-value1
    key-name2=key-value2

    [section-name2]
    key-name3=key-value3
    key-name4=key-value4

    這種配置文件中的參數以 section 分組,然后用 key=value 的形式指定參數及其值。應用程序也可以利用這種配置文件格式保存一些配置信息,為此,MiniGUI 提供了如下三個函數(include/minigui.h):


    int GUIAPI GetValueFromEtcFile (const char* pEtcFile, const char* pSection,const char* pKey, char* pValue, int iLen);
    int GUIAPI GetIntValueFromEtcFile (const char* pEtcFile, const char* pSection,const char* pKey, int* value);
    int GUIAPI SetValueToEtcFile (const char* pEtcFile, const char* pSection, const char* pKey, char* pValue);

    這三個函數的用途如下:

    • GetValueFromEtcFile:從指定的配置文件當中獲取指定的鍵值,鍵值以字符串形式返回。
    • GetIntValueFromEtcFile:從指定的配置文件當中獲取指定的整數型鍵值。該函數將獲得的字符串轉換為整數值返回(采用strtol 函數轉換)。
    • SetValueToEtcFile:該函數將給定的鍵值保存到指定的配置文件當中,如果配置文件不存在,則將新建配置文件。如果給定的鍵已存在,則將覆蓋舊值。

    假定某個配置文件記錄了一些應用程序信息,并具有如下格式:


    [mginit]
    nr=8
    autostart=0

    [app0]
    path=../tools/
    name=vcongui
    layer=
    tip=Virtual&console&on&MiniGUI
    icon=res/konsole.gif

    [app1]
    path=../bomb/
    name=bomb
    layer=
    tip=Game&of&Minesweaper
    icon=res/kmines.gif

    [app2]
    path=../controlpanel/
    name=controlpanel
    layer=
    tip=Control&Panel
    icon=res/kcmx.gif

    其中的 [mginit] 段記錄了應用程序個數(nr鍵),以及自動啟動的應用程序索引(autostart鍵)。而 [appX] 段記錄了每個應用程序的信息,包括該應用程序的路徑、名稱、圖標等等。下面的代碼演示了如何使用 MiniGU的配置文件函數獲取這些信息(該代碼段來自 mde 演示包中的 mginit 程序):


    #define APP_INFO_FILE "mginit.rc"

    static BOOL get_app_info (void)
    {
    int i;
    APPITEM* item;

    /* 獲取應用程序個數信息 */
    if (GetIntValueFromEtcFile (APP_INFO_FILE, "mginit", "nr", &app_info.nr_apps) != ETC_OK)
    return FALSE;

    if (app_info.nr_apps <= 0)
    return FALSE;

    /* 獲取自動啟動的應用程序索引 */
    GetIntValueFromEtcFile (APP_INFO_FILE, "mginit", "autostart", &app_info.autostart);

    if (app_info.autostart >= app_info.nr_apps || app_info.autostart < 0)
    app_info.autostart = 0;

    /* 分配應用程序信息結構 */
    if ((app_info.app_items = (APPITEM*)calloc (app_info.nr_apps, sizeof (APPITEM))) == NULL) {
    return FALSE;
    }

    /* 獲取每個應用程序的路徑、名稱、圖標等信息 */
    item = app_info.app_items;
    for (i = 0; i < app_info.nr_apps; i++, item++) {
    char section [10];

    sprintf (section, "app%d", i);
    if (GetValueFromEtcFile (APP_INFO_FILE, section, "path", item->path, PATH_MAX) != ETC_OK)
    goto error;

    if (GetValueFromEtcFile (APP_INFO_FILE, section, "name", item->name, NAME_MAX) != ETC_OK)
    goto error;

    if (GetValueFromEtcFile (APP_INFO_FILE, section, "layer", item->layer, LEN_LAYER_NAME) != ETC_OK)
    goto error;

    if (GetValueFromEtcFile (APP_INFO_FILE, section, "tip", item->tip, TIP_MAX) != ETC_OK)
    goto error;

    strsubchr (item->tip, '&', ' ');

    if (GetValueFromEtcFile (APP_INFO_FILE, section, "icon", item->bmp_path, PATH_MAX + NAME_MAX) != ETC_OK)
    goto error;

    if (LoadBitmap (HDC_SCREEN, &item->bmp, item->bmp_path) != ERR_BMP_OK)
    goto error;

    item->cdpath = TRUE;
    }
    return TRUE;

    error:
    free_app_info ();
    return FALSE;
    }

    5.2 定點數運算
    通常在進行數學運算時,我們采用浮點數表示實數,并利用 頭文件中所聲明的函數進行浮點數運算。我們知道,浮點數運算是一種非常耗時的運算過程。為了減少因為浮點數運算而帶來的額外 CPU 指令,在一些三維圖形庫當中,通常會采用定點數來表示實數,并利用定點數進行運算,這樣,將大大提高三維圖形的運算速度。MiniGUI 也提供了一些定點數運算函數,分為如下幾類:

    • 整數、浮點數和定點數之間的轉換。利用 itofix 和 fixtoi 函數可實現整數和定點數之間的相互轉換;利用 ftofix 和 fixtof 函數可實現浮點數和定點數之間的轉換。
    • 定點數加、減、乘、除等基本運算。利用 fadd、fsub、fmul、fdiv、fsqrt等函數可實現定點數加、減、乘、除以及平方根運算。
    • 定點數的三角運算。利用 fcos、fsin、ftan、facos、fasin 等函數可求給定定點數的余弦、正弦、正切、反余弦、反正弦值。
    • 矩陣、向量等運算。矩陣、向量相關運算在三維圖形中非常重要,限于篇幅,本文不會詳細講述這些運算,讀者可參閱MiniGUI 的 include/fixedmath.h 頭文件。

    下面的代碼段演示了定點數的用法,該程序段根據給定的三個點(pts[0]、pts[1]、pts[2])畫一個弧線,其中 pts[0] 作為圓心,pts[1] 是圓弧的起點,而 pts[2] 是圓弧終點和圓心連線上的一個點:


    void draw_arc (HDC hdc, POINT* pts)
    {
    int sx = pts [0].x, sy = pts [0].y;
    int dx = pts [1].x - sx, dy = pts [1].y - sy;
    int r = sqrt (dx * dx * 1.0 + dy * dy * 1.0);
    double cos_d = dx * 1.0 / r;
    fixed cos_f = ftofix (cos_d);
    fixed ang1 = facos (cos_f);
    int r2;
    fixed ang2;

    if (dy > 0) {
    ang1 = fsub (0, ang1);
    }

    dx = pts [2].x - sx;
    dy = pts [2].y - sy;
    r2 = sqrt (dx * dx * 1.0 + dy * dy * 1.0);
    cos_d = dx * 1.0 / r2;
    cos_f = ftofix (cos_d);
    ang2 = facos (cos_f);
    if (dy > 0) {
    ang2 = fsub (0, ang2);
    }

    Arc (hdc, sx, sy, r, ang1, ang2);
    }

    上述程序的計算非常簡單,步驟如下(該程序段來自 mde 演示程序包中的 painter/painter.c 程序):

  • 根據 pts[0] 和 pts[1] 計算圓弧的半徑,然后計算圓弧的起始偏角,即 ang1,使用了ftofix 函數和 facos 函數。
  • 計算 pts[2] 點和圓心連線的夾角,即 ang2,使用了 ftofix 和 facos 函數。
  • 調用 Arc 函數繪制圓弧。
  • 6 小結
    本文講述了 MiniGUI 為應用程序提供的一些非 GUI/GDI 的接口。這些接口中,某些是為了解決和操作系統的交互而設計的,以便 MiniGUI 應用程序能夠更好地與操作系統提供的機制融合在一起;而某些提供了對 UNIX Domain Socket 良好封裝的接口,可幫助應用程序方便進行進程間通訊或者擴展其功能;其他接口則專注于嵌入式系統的特殊性,為應用程序提供了可移植的文件 I/O 封裝代碼。在這些接口的幫助下,嵌入式系統開發人員可以編寫功能強大而靈活的應用程序。

    轉載于:https://www.cnblogs.com/weifuqin530/archive/2009/05/17/1458628.html

    總結

    以上是生活随笔為你收集整理的基于 Linux 和 MiniGUI 的嵌入式系统软件开发指南(六) MiniGUI 提供的非 GUI/GDI 接口...的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    久久久黄色av| 国产免费人人看 | 亚洲三级在线免费观看 | 国产成人精品亚洲日本在线观看 | 一级黄色片在线免费看 | 精品视频99 | 福利网在线 | 成人免费网站视频 | 欧美成人理伦片 | 国产一区高清在线 | 亚洲黄色免费网站 | 天天射日 | 中文字幕亚洲在线观看 | 色综合天天色 | 91视频在线看 | 丁香色婷婷 | 国产成人福利在线 | 久久久久在线观看 | 99热亚洲精品 | av在线一级 | 国产高清中文字幕 | 国产精品一区二区三区在线播放 | 久热国产视频 | 久久99精品热在线观看 | 午夜色大片在线观看 | 欧美在线视频精品 | 欧美精品久久久久久久久久 | 91成人免费看 | 国产视频在 | 亚洲天堂色婷婷 | 五月天久久 | 成年人免费观看在线视频 | 欧美日韩中文国产一区发布 | 欧美一区二区三区在线看 | 国产亚洲永久域名 | 亚洲欧美综合 | 91av在线免费视频 | 九九涩涩av台湾日本热热 | 在线免费高清一区二区三区 | 天天色天天射天天综合网 | 激情综合色综合久久 | 成年一级片 | 日韩免费一级电影 | 中文字幕免费看 | 狠狠狠色丁香婷婷综合久久88 | 免费在线观看av片 | 亚洲japanese制服美女 | 最新动作电影 | 亚洲国产精品500在线观看 | 国产 字幕 制服 中文 在线 | 欧美动漫一区二区三区 | 日韩精品在线免费播放 | 亚洲蜜桃av | 国产精品久久久久影院日本 | 日批视频在线观看免费 | 精品影院一区二区久久久 | 国产精品免费在线视频 | 久久久在线视频 | av亚洲产国偷v产偷v自拍小说 | 亚洲精品毛片一级91精品 | 国产黄在线免费观看 | 成人aⅴ视频 | 国产黄色一级片 | 在线精品视频免费播放 | 国产啊v在线观看 | 91视频 - v11av| 色噜噜噜噜 | 在线 视频 亚洲 | 天天操天天草 | 天天操伊人| 欧美精品久久久久久久久久 | www.888av| 五月婷婷综合在线视频 | 日韩在线观看网站 | 国产精品午夜久久 | 中文字幕网站视频在线 | 国产精品久久久久久久av大片 | 青青河边草免费直播 | 9ⅰ精品久久久久久久久中文字幕 | 国内精品久久久久久久影视麻豆 | 激情 一区二区 | 免费视频久久久久 | 免费在线看成人av | 特片网久久| 国产午夜精品免费一区二区三区视频 | 成年美女黄网站色大片免费看 | av福利电影 | 欧美大片大全 | av国产网站 | 免费精品视频在线 | 亚洲精品免费在线观看视频 | 色天天综合网 | 亚洲视频一区二区三区在线观看 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 日日干干夜夜 | 亚洲午夜av久久乱码 | 婷婷在线观看视频 | 天天爽天天爽天天爽 | 精品99999 | 六月婷婷久香在线视频 | 欧美视频18 | 91精选在线观看 | 亚洲专区在线 | 久久国产精品免费一区 | 97在线视频免费 | 成人黄色小说在线观看 | 成人av久久| 色综合天 | 99色在线观看 | 国产香蕉久久精品综合网 | 99久久婷婷 | www.日本色 | 人人玩人人添人人澡97 | 日韩视频免费观看高清 | 久久久国产一区二区三区四区小说 | 中文字幕黄色网 | 欧美日韩国产精品一区二区亚洲 | 又黄又爽免费视频 | 97碰在线视频 | 亚洲成人高清在线 | 青青久草在线 | 黄色软件在线观看免费 | 成人网看片| 伊人成人久久 | 五月婷婷av | 99精品视频在线观看播放 | 黄色av网站在线观看免费 | 亚洲成人黄色在线观看 | 天天干天天干天天干天天干天天干天天干 | 国产成人精品久久亚洲高清不卡 | 欧美少妇xxxxxx | 麻豆久久久久久久 | 最近中文字幕视频网 | 久久久精品免费观看 | 亚洲精品自拍视频在线观看 | 91探花视频 | 欧美最猛性xxxxx免费 | 国产精品一区二区在线免费观看 | 伊人国产女 | 中文字幕观看av | 色诱亚洲精品久久久久久 | 9i看片成人免费看片 | 亚洲天堂精品 | 免费涩涩网站 | 日本不卡一区二区三区在线观看 | 国产999精品久久久 免费a网站 | 91色影院 | 男女拍拍免费视频 | 日韩艹| 国产美腿白丝袜足在线av | 成 人 黄 色 免费播放 | 日韩综合色 | 性色av免费在线观看 | 黄网站免费看 | 国产精品视频久久 | 成人久久18免费网站 | 久久夜夜操 | 久久,天天综合 | 国内精品久久久久 | 97精品国产91久久久久久 | 国产午夜精品一区二区三区在线观看 | free,性欧美 九九交易行官网 | 国产69久久精品成人看 | 亚在线播放中文视频 | 国产一区二区免费在线观看 | 国产福利免费看 | 久久在线观看 | 日韩理论片中文字幕 | 欧美日韩中文在线 | 婷婷av网 | 91久久奴性调教 | 97精品久久 | av在线免费观看黄 | av专区在线 | 夜夜操狠狠操 | 国产精品第一 | 久久精品99北条麻妃 | 看av在线| 久久这里只有精品23 | 久精品视频在线 | 中文字幕在线观看三区 | 免费aa大片| 久久久久 | 成年人电影毛片 | 久久天天躁夜夜躁狠狠躁2022 | 亚洲首页 | 久久综合狠狠综合久久激情 | 国产无套一区二区三区久久 | 久久免费看 | 精品天堂av| 国产精品欧美久久久久三级 | 天天曰天天爽 | 亚州av成人| 91精品蜜桃 | 亚洲精品黄| 8x成人免费视频 | 女人久久久久 | 丁香午夜婷婷 | 日韩欧美一区视频 | 99精品视频在线观看 | 91精品国产麻豆 | 国产精品久久久久久久av电影 | 亚洲在线不卡 | 激情五月网站 | 波多野结衣一区二区三区中文字幕 | 欧美性生交大片免网 | 免费黄色av电影 | 国产精品欧美久久久久无广告 | 国产精品毛片久久久久久久久久99999999 | 午夜黄色| 天天玩天天操天天射 | 亚洲精品乱码久久久久久按摩 | 免费看的黄色的网站 | 在线观看视频99 | 麻豆视频免费在线 | 亚洲成av人片一区二区梦乃 | 亚洲视频在线看 | 日韩视频一区二区三区 | 日韩高清网站 | 国产黄色免费电影 | 日韩在线第一区 | 午夜国产在线观看 | av电影久久 | 精品国产伦一区二区三区观看方式 | 国产免费视频一区二区裸体 | 国产69精品久久app免费版 | 日韩高清免费电影 | 国产成人黄色 | 九九热在线观看 | 99re6热在线精品视频 | 久久免费在线观看视频 | 亚洲永久精品在线 | 国产91精品看黄网站 | 日日爱网站 | 在线免费av网站 | 九九热99视频 | 久久国产美女 | 91片黄在线观 | 欧美污网站| 特级黄录像视频 | 色婷婷天天干 | 国产精品区一区 | 伊人中文字幕在线 | 欧美国产日韩一区二区三区 | 久久精品国产精品 | 激情偷乱人伦小说视频在线观看 | 亚洲精品视频在线观看免费视频 | 麻豆影视在线播放 | 日本黄网站 | 亚洲国产成人精品在线 | 久草在线免 | 九九免费精品视频 | 狠狠干夜夜操 | 人人搞人人搞 | 久久久久久蜜av免费网站 | 国产精品第一 | 福利精品在线 | 成人免费视频播放 | 国产精品一区二区在线 | 精品国产一区二区三区日日嗨 | 成年人免费在线观看 | 亚洲免费av在线播放 | 91试看| 日韩城人在线 | 久草视频网 | 国产v欧美| 美女视频黄免费网站 | 国产免费中文字幕 | 国产人成在线视频 | 亚洲视频 一区 | 精品视频国产 | 96久久精品 | 欧美午夜激情网 | 日韩中文在线观看 | 超碰97人 | 日本精品久久久久影院 | 免费a v视频 | 亚洲精品乱码白浆高清久久久久久 | 欧美日韩性视频 | 亚洲国产偷 | 国产亚洲欧美日韩高清 | 久久久夜色 | 久久久精品一区二区三区 | 久久久国产精华液 | 91精品国产福利在线观看 | 久久99网| 精品日韩在线 | 欧美精品少妇xxxxx喷水 | 九九激情视频 | 国产一区二区在线观看视频 | 国产裸体永久免费视频网站 | 久久66热这里只有精品 | 91九色蝌蚪视频网站 | 久久精品国产久精国产 | 草樱av| 国产精品免费一区二区三区在线观看 | 亚洲精品黄色在线观看 | 国产一区网 | 国产免费观看久久 | 国产成人精品午夜在线播放 | 成人黄色在线观看视频 | 久久综合久久综合久久综合 | 亚洲尺码电影av久久 | 激情综合一区 | 国产视频 亚洲视频 | 国产精品大片免费观看 | 中午字幕在线观看 | 天天摸天天舔天天操 | 激情婷婷丁香 | 精品国产亚洲在线 | 日本aaaa级毛片在线看 | 91国内在线 | 国产在线97 | 国产麻豆果冻传媒在线观看 | 免费韩国av | 五月天久久久久久 | 精品久久免费看 | 美女视频黄网站 | 国产69久久久 | 天天射天天 | 亚洲综合激情 | 狠色在线 | 免费观看黄色12片一级视频 | 免费在线观看一区二区三区 | 色小说av | 麻花天美星空视频 | 美女视频黄,久久 | 国产精品va视频 | 国产精品久久久久久久久久久免费 | 亚洲美女视频在线 | 在线观看视频97 | 免费视频久久久久久久 | 国产99精品在线观看 | 国产在线 一区二区三区 | 国产亚洲精品美女 | 91久久精品一区二区二区 | 精品国产一区二 | 91中文字幕永久在线 | 久久 精品一区 | 人人爽人人爽人人爽人人爽 | 9999精品免费视频 | 91在线视频免费观看 | 亚洲a成人v | 国产偷v国产偷∨精品视频 在线草 | 黄色三几片| 97夜夜澡人人爽人人免费 | 久久在线视频在线 | 国内综合精品午夜久久资源 | 久久乐九色婷婷综合色狠狠182 | 中文字幕一区二区在线观看 | 国产精品18久久久久vr手机版特色 | 久久久久国产免费免费 | 国产亚洲资源 | 97在线看| 亚洲国产大片 | 免费日韩一区二区三区 | 日日夜夜天天人人 | 97超视频免费观看 | 一本一道久久a久久精品蜜桃 | 日日摸日日碰 | 午夜狠狠操 | 99精品视频精品精品视频 | 国产精品九九热 | 91爱爱视频 | 草久在线 | 啪啪动态视频 | 日韩欧美电影在线 | 久久久久成人精品免费播放动漫 | 五月婷av | 久久婷婷一区二区三区 | 精品在线观看视频 | 在线视频 91 | 国产精品一区二区三区四区在线观看 | 欧美一级片播放 | 日韩免费在线观看视频 | 91插插影库 | 91在线日韩 | 久久久精品久久 | 99在线视频网站 | 欧美日韩免费在线视频 | 欧美在线视频一区二区三区 | 在线视频 精品 | 91丨九色丨蝌蚪丨对白 | 91av国产视频 | 麻豆超碰| 国产九色91| 日本久久精 | 天天草天天干 | 狠狠色噜噜狠狠狠狠 | 色综合久久久久久久 | 99久久er热在这里只有精品66 | 青草视频在线播放 | 中文字幕色综合网 | 亚洲精品国产麻豆 | 欧美成人高清 | 色福利网站 | 成人免费共享视频 | 五月婷婷综合在线视频 | 一区 二区电影免费在线观看 | 不卡国产视频 | 国产精品久久久久9999 | 亚洲激情 欧美激情 | 国产成人久久精品77777 | 玖玖玖在线观看 | 日日精品 | 欧美在线1| 蜜臀av夜夜澡人人爽人人桃色 | 亚洲国产午夜 | 国产成人精品网站 | 免费观看一级视频 | 亚洲成av人片在线观看香蕉 | 丁香六月婷婷开心婷婷网 | 人人搞人人干 | 91片黄在线观看 | 高清国产午夜精品久久久久久 | 国产午夜精品av一区二区 | 成人在线免费小视频 | www.夜夜操.com | 91成人观看| 亚洲天堂网在线视频 | 久久精品区 | 激情片av | 天天色天天综合网 | 91亚洲成人 | 在线观看视频99 | 久在线观看| 久草免费资源 | 欧美先锋影音 | 在线看的av网站 | 97视频在线观看网址 | 91精品1区2区 | 亚洲成人精品国产 | 成人在线免费视频 | 欧美另类色图 | 久久精品综合视频 | 日韩色中色 | 国产视频1区2区3区 久久夜视频 | 国产精品一区二区三区免费看 | 国产一级一片免费播放放 | 麻豆视屏| 99视频国产精品免费观看 | 精品国内自产拍在线观看视频 | 二区三区在线视频 | 亚洲精品免费在线播放 | 天天插夜夜操 | 中文字幕在线电影 | 九九久久免费视频 | 人人添人人澡人人澡人人人爽 | 久久国产精品久久国产精品 | 国产区久久| 日韩三级视频在线看 | 日韩有码欧美 | 99久久久久成人国产免费 | 免费看黄网站在线 | 日韩高清精品免费观看 | 99精品热| 国产91学生粉嫩喷水 | 免费看国产a | 国产中文字幕视频在线观看 | 久久综合一本 | 国产三级精品在线 | 99免在线观看免费视频高清 | 一区二区三区在线视频111 | 欧美激情视频一区二区三区 | 国产又粗又猛又色又黄视频 | 亚洲国产日韩欧美在线 | 99在线观看免费视频精品观看 | 成人 亚洲 欧美 | 免费av网站在线看 | 国产99免费视频 | 美女网站视频久久 | 中文字幕日本在线观看 | 婷婷狠狠操 | 美女视频黄在线 | 色婷婷色 | 婷婷丁香狠狠爱 | 国产高清中文字幕 | 亚洲黄色一级电影 | 国产一区二区在线播放视频 | 97视频网站| 久久久综合精品 | 日日夜夜艹 | 国产123区在线观看 国产精品麻豆91 | 亚洲伦理一区二区 | 成人黄色电影在线 | 成人四虎 | 一本一本久久a久久精品综合妖精 | 亚洲精品综合一二三区在线观看 | 国产精品成久久久久三级 | 免费福利片2019潦草影视午夜 | 免费看黄电影 | 日韩精品中文字幕有码 | 日韩欧美xxxx | 日本性视频 | 欧美孕妇与黑人孕交 | 91久久国产自产拍夜夜嗨 | 中文字幕在线观看亚洲 | 日韩美在线观看 | 91亚洲精品视频 | 亚洲激情| 天天射,天天干 | 免费视频久久久久 | 久久99国产一区二区三区 | 日韩夜夜爽 | 国产香蕉视频在线播放 | a黄色一级 | 粉嫩av一区二区三区四区 | 日日综合网| 国产视频二 | 伊人天堂网| 亚洲精品日韩av | 草在线 | 国产第一页在线观看 | 欧美在线视频一区二区三区 | 粉嫩av一区二区三区四区五区 | 久久国际影院 | 国产在线欧美在线 | 日韩免费看片 | 欧美国产不卡 | 一区二区三区电影在线播 | av网站在线免费观看 | aaa日本高清在线播放免费观看 | 国产精品综合av一区二区国产馆 | 最新日韩在线观看 | 国产精品九九热 | 五月开心网 | 亚洲亚洲精品在线观看 | 欧美日韩中文在线观看 | 国产精品一区免费在线观看 | 中文字幕频道 | 麻豆视频一区二区 | 丰满少妇一级 | 一区二区三区日韩视频在线观看 | 成人av网站在线播放 | 成人黄色在线看 | 三级黄色网址 | 免费网站黄 | 国产综合在线观看视频 | 超碰97.com | 欧美日韩亚洲精品在线 | 成人三级网站在线观看 | 夜夜躁天天躁很躁波 | 婷婷久久综合九色综合 | 日韩av一区二区三区在线观看 | 天干啦夜天干天干在线线 | 99视频一区二区 | 成人a在线观看高清电影 | 婷婷狠狠操 | 99精品亚洲 | 一级淫片a | 天天操天天曰 | 91麻豆免费版 | 夜添久久精品亚洲国产精品 | 中文字幕丝袜一区二区 | 国产黄色a| 在线中文字幕一区二区 | 在线成人免费电影 | 中文字幕av专区 | 色婷婷精品 | 在线播放亚洲激情 | 午夜精品av| 色综合久久久久综合体 | 欧美日韩免费在线观看视频 | 午夜电影久久久 | www四虎影院 | 成人黄色在线视频 | 在线a亚洲视频播放在线观看 | 91精品少妇偷拍99 | 欧美激情精品久久久久久 | a天堂在线看| 成片视频在线观看 | 黄色在线看网站 | 91天堂影院 | 久久综合九色综合97婷婷女人 | 麻豆91在线播放 | 97超碰免费 | 精品播放 | 五月激情丁香 | 精品亚洲免费视频 | 午夜精品一区二区三区在线 | 日韩精品免费一区二区在线观看 | 人人玩人人添人人澡超碰 | 免费看污片 | 国产成人综合图片 | 在线色亚洲 | 99久久精品国产免费看不卡 | 91精品国产乱码久久桃 | 日韩av在线资源 | 91免费观看视频网站 | 婷婷丁香国产 | 国产中文字幕三区 | 1000部18岁以下禁看视频 | 久久精品免费看 | 久久久久北条麻妃免费看 | 91成人精品 | 黄网站色视频 | 成人aaa毛片 | 日韩电影中文字幕在线 | 国产精品一区二区三区在线免费观看 | 中文字幕在线看 | 日韩网站在线播放 | 久久久亚洲影院 | 亚洲国产成人高清精品 | 九九久久免费视频 | 日韩三级久久 | 亚洲一区二区三区四区精品 | 91在线小视频| 91成人网在线播放 | 国产不卡一区二区视频 | 国产综合精品久久 | 国产成人精品一区二区三区福利 | 婷婷精品进入 | 国产精品毛片一区二区在线 | 久久在线免费视频 | 99视频精品全国免费 | 欧美日韩一区二区三区在线免费观看 | 日韩精品免费一区二区 | 国产精品av在线免费观看 | 国产一级在线播放 | 久草在线观 | 三级视频片 | 色偷偷网站视频 | 伊人六月 | 亚洲精品综合一区二区 | 成人在线视频论坛 | 日韩精品综合在线 | 亚洲精品乱码久久久久久按摩 | 国产精品露脸在线 | 五月激情丁香婷婷 | 九七在线视频 | 国产真实在线 | 欧美黄色软件 | 欧美午夜精品久久久久久孕妇 | 国产福利一区二区三区视频 | 狠色狠色综合久久 | 高清不卡毛片 | 精品一区久久 | 狠狠的操你 | 国产色女人 | 91精品久久久久久综合乱菊 | 欧美一进一出抽搐大尺度视频 | 欧美激情视频在线观看免费 | 少妇精品久久久一区二区免费 | 国产在线小视频 | 成年人国产在线观看 | 2023亚洲精品国偷拍自产在线 | 99精品在线免费在线观看 | 国产在线a不卡 | avwww在线观看| 精品视频www | 国产精品免费看久久久8精臀av | 国产福利91精品 | 黄色三级av| 午夜三级理论 | 美女国产在线 | 综合国产视频 | 婷婷视频导航 | 国产综合精品久久 | 国产福利91精品 | 国产精品视频免费在线观看 | 国产免费观看高清完整版 | av免费试看 | 久久精品视频99 | 天天透天天插 | www.夜夜操.com| 最近高清中文在线字幕在线观看 | 91麻豆看国产在线紧急地址 | 精品免费久久 | 色视频网站在线观看一=区 a视频免费在线观看 | 亚洲人片在线观看 | 精品国产资源 | 天天操狠狠操夜夜操 | 韩国av一区 | 国产精品毛片一区二区 | 激情av一区二区 | 黄色日视频 | 97电影网站| 波多野结衣最新 | 香蕉视频在线看 | 日韩欧美区| 成人午夜电影在线播放 | 亚洲在线资源 | 六月色婷| 国产天天爽 | 开心激情久久 | 久草a在线| 国产精品视频不卡 | 国产精品观看在线亚洲人成网 | av资源在线看| 99视频久| 婷婷爱五月天 | 欧美精品v国产精品v日韩精品 | www.黄色小说.com | 国产高清无线码2021 | 日本三级久久 | 国产最新视频在线观看 | 丝袜美女视频网站 | 欧美伦理一区 | 久久96国产精品久久99漫画 | 国产亚洲婷婷免费 | 91成人免费电影 | 高清av网站 | 四虎8848免费高清在线观看 | 亚洲精选视频在线 | 亚洲精品一区二区在线观看 | 伊在线视频 | 国产专区日韩专区 | 日韩在线视频国产 | www.五月天激情 | 亚洲成人精品国产 | 尤物九九久久国产精品的分类 | 在线黄色毛片 | 99国产精品视频免费观看一公开 | 亚洲精品欧美成人 | 日韩精品一区二区三区高清免费 | 99热这里只有精品8 久久综合毛片 | 中文字幕999| 91麻豆福利 | 天堂在线v | 激情av一区二区 | 国产在线观看地址 | 日韩中文字幕在线 | 精品国产免费人成在线观看 | 中文字幕av全部资源www中文字幕在线观看 | 国产精品夜夜夜一区二区三区尤 | 综合网久久| 成人a视频片观看免费 | 在线不卡中文字幕播放 | 亚洲国产伊人 | www夜夜操| 欧美午夜理伦三级在线观看 | 尤物九九久久国产精品的分类 | 操操操人人 | 在线欧美中文字幕 | 97精品视频在线播放 | av资源网在线播放 | 在线观看视频一区二区 | 亚洲无吗视频在线 | 91视频免费网站 | 日韩欧美在线综合网 | 日韩手机视频 | 国产在线超碰 | 蜜桃麻豆www久久囤产精品 | 日韩免费区 | 日韩免费电影在线观看 | 成人精品久久久 | 亚洲一区免费在线 | 天天天天天操 | 91在线一区| 国产91学生粉嫩喷水 | 久久国产精品视频 | 97电影手机 | 永久av免费在线观看 | 99久久超碰中文字幕伊人 | 国产视频亚洲视频 | 成人久久久电影 | 波多野结衣在线播放一区 | 在线观看 国产 | 日韩mv欧美mv国产精品 | 亚洲精品久久久久www | 久草影视在线观看 | 国产群p视频 | 国色天香在线 | 91视频午夜| 国产亚洲永久域名 | 精品国产精品久久一区免费式 | 91自拍成人| 午夜精品一区二区三区免费 | 中文字幕有码在线 | 中文字幕在线免费观看 | 成年人视频免费在线播放 | 久久情侣偷拍 | 超碰在线色 | 精品亚洲一区二区 | 国产精品门事件 | 91成人在线观看高潮 | 亚洲综合色视频 | 中文字幕av在线播放 | 97视频人人澡人人爽 | 成人免费影院 | 亚洲激情在线观看 | av韩国在线| av三级在线免费观看 | 久久精品www人人爽人人 | 欧美日韩国产页 | 亚州精品在线视频 | 永久免费在线 | 伊人宗合网 | 欧美国产在线看 | 色在线视频网 | 国产专区一 | 一区二区视频在线看 | 狠狠做深爱婷婷综合一区 | www.亚洲在线| 91av视频免费观看 | 99久久精品免费看国产一区二区三区 | 中文字幕日韩在线播放 | 91免费的视频在线播放 | 久久再线视频 | 在线国产欧美 | 在线观看免费日韩 | 日韩中文字幕一区 | 一区二区三区在线观看 | 日韩在线免费观看视频 | 国产精品婷婷午夜在线观看 | 午夜精品一区二区国产 | 免费在线观看av的网站 | 国产精品久久久久久久久久尿 | 美女网站视频免费黄 | 91精品国产高清自在线观看 | 在线观看精品黄av片免费 | av免费看电影 | 亚洲国产精品激情在线观看 | 99久久精品国产免费看不卡 | 欧洲色综合 | 在线超碰av | 日韩专区在线播放 | 中文字幕在线日本 | 欧美精品乱码久久久久久按摩 | 久久再线视频 | 婷婷成人在线 | 91精品一区二区在线观看 | 婷婷激情网站 | 亚洲一区视频免费观看 | 日本一区二区高清不卡 | 成人精品99 | 欧美精品乱码99久久影院 | 亚洲成人在线免费 | 国产精品爽爽久久久久久蜜臀 | 中文字幕4 | 不卡的av电影 | 狠狠色丁香久久婷婷综合_中 | 91香蕉视频在线下载 | 国产精品12345 | 日本高清xxxx| 色婷婷婷 | 狠狠色狠狠色综合系列 | 国产在线va | 成人av片免费看 | 亚洲精品久久久久久久蜜桃 | 日韩一区二区三区高清免费看看 | 欧美亚洲国产一卡 | 91免费日韩 | 超级碰99| 国产精品视频在线观看 | 婷婷av网 | 国产一级片毛片 | 亚洲精品小视频 | 国产精品一区二区av影院萌芽 | 中文字幕丝袜 | 亚洲欧美日韩精品久久奇米一区 | 九九久久久 | 国产一区二区视频在线播放 | 国产小视频在线免费观看 | 深夜免费福利视频 | 天天色天天综合 | 麻豆免费视频网站 | 天天躁日日躁狠狠躁av麻豆 | 国产福利一区二区三区视频 | 国产精品热视频 | 天天干.com| 一区在线观看 | 久久精品视频一 | 国产精品青青 | 黄视频网站大全 | 日本资源中文字幕在线 | 国产亚洲精品日韩在线tv黄 | 99视频在线免费看 | 国产成人99久久亚洲综合精品 | 久久天堂网站 | 99综合影院在线 | 久久精品男人的天堂 | 日韩黄色在线观看 | 香蕉成人在线视频 | 一区二区欧美日韩 | 射久久久| 日韩亚洲国产精品 | 精品黄色片 | 91污在线| 国产精品久久久久久久久久 | 欧美高清视频不卡网 | 黄av在线| 国产 欧美 日本 | 色噜噜狠狠色综合中国 | www免费网站在线观看 | 人人看黄色 | 国产91精品久久久久 | 亚洲女人天堂成人av在线 | 成人免费视频网站 | 亚洲精选在线观看 | 精品欧美一区二区三区久久久 | 亚洲综合涩 | 欧美精品三级 | 欧美一区二区三区免费观看 | 天天操天天色天天 | 免费观看91视频大全 | 国产在线最新 | 97人人模人人爽人人喊中文字 | 正在播放国产精品 | 国产又粗又长又硬免费视频 | 亚洲视频一区二区三区在线观看 | 伊人婷婷久久 | 成人午夜黄色影院 | 麻豆精品视频在线观看免费 | 久久免费视频观看 | 国产三级午夜理伦三级 | 国产亚洲精品日韩在线tv黄 | 久久久久久不卡 | 国产中文视频 | 九九九视频精品 | 国产精品激情偷乱一区二区∴ | 国产一级在线免费观看 | 美女久久 | 一区二区三区在线电影 | 久久国产经典 | 国产91学生粉嫩喷水 | 亚洲欧美在线视频免费 | 色吊丝av中文字幕 | 成年人免费av网站 | 久久公开视频 | 91福利区一区二区三区 | 中文乱码视频在线观看 | 欧美一区二区三区不卡 | 国产精品美女999 | 五月婷网站 | 天天操天天操天天操天天操天天操 | 黄色官网在线观看 | 国产精品成人一区二区三区 | 午夜精品久久 | 久久久在线视频 | 五月天亚洲综合小说网 | 欧美日韩性视频在线 | 韩国av永久免费 | 久久久av电影 | 亚洲精品视频一二三 | 69久久夜色精品国产69 | 亚洲午夜精品一区 | 国产激情小视频在线观看 | 国产美女黄网站免费 | 国产午夜精品一区二区三区 | 992tv在线观看 | www夜夜| 国产亚洲精品久久久久久久久久 | 一二区av | 国产午夜在线观看视频 | 日本丶国产丶欧美色综合 | 婷婷色影院 | 黄色小网站在线观看 | 免费在线观看av网址 | 精品免费一区二区三区 | 精品久久久久久久 | 亚洲精品自在在线观看 | 91av视频在线播放 | 国产精品日韩高清 | 精品国产一区二区久久 | 亚洲第一中文网 | 精品亚洲视频在线观看 | 成人精品福利 | 国产精品 9999 | av黄免费看| 天天射天天操天天 | 99精品视频免费观看视频 | www.成人精品| 亚洲国产精品成人va在线观看 | 久久黄色影院 | 国产视频亚洲 | 国产精品18久久久久白浆 | 久久精品国产第一区二区三区 | 美女精品在线 | 免费在线国产视频 | 亚洲国产欧洲综合997久久, | 久久国产精品久久久 | 欧美少妇的秘密 | 激情偷乱人伦小说视频在线观看 | 国产视频在线免费 | 天天操夜夜拍 | 97在线视频免费 | 免费看污污视频的网站 | 激情五月伊人 | 国产黄色在线观看 | 亚洲午夜小视频 | 国产伦精品一区二区三区四区视频 | 麻豆久久一区 | 一区二区三区在线视频111 | 成人午夜电影免费在线观看 | 久久综合毛片 | 黄色成人小视频 | 91精品在线免费 | 人人藻人人澡人人爽 | 狠狠色噜噜狠狠 | 亚洲闷骚少妇在线观看网站 | 人人爽人人舔 | 黄色小视频在线观看免费 | av电影在线免费 | 日韩欧美在线观看 | 日韩在线不卡视频 | 国产18精品乱码免费看 | 在线观看涩涩 |