青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

string

string
posts - 27, comments - 177, trackbacks - 0, articles - 0
  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

UEFI 實戰(4) protocol

Posted on 2012-03-06 07:30 djx_zh 閱讀(18181) 評論(18)  編輯 收藏 引用
什么是protocol
從字面意思上看,protocol是server和client之間的一種約定,雙方根據這種約定互通信息。這里的server和client是一種廣義的稱呼,提供服務的稱為server,使用服務的稱為client。 TCP是一種protocol, client(應用程序)通過一組函數來壓包和解包,壓包和解包是server提供的服務。COM也是一種protocol,client通過CoCreateInstance(...)和GUID獲得指向COM對象的指針,然后使用該指針獲得COM對象提供的服務, GUID標示了這個COM對象?,F在我們對protocol有了概念上的理解,那么具體到UEFI里,protocol是什么樣子呢? 如何標示一個protocol?如何得到protocol對應的對象?...容我慢慢道來.
在講protocol什么樣子之前,還要插幾句C與C++的區別。我們知道UEFI是用C來開發的,C是面向過程的一種語言。而管理和使用UEFI眾多的protocol完全使用面向過程的思想會使程序變得復雜。protocol作為一種對象來設計管理會比較直觀。因而UEFI中的Protocol引入了面向對象的思想,用struct來模擬class, Protocol用struct來實現,用函數指針(Protocol的成員變量)模擬成員函數,此種函數的第一參數必須是指向Protocol的指針,用來模擬this指針。
Protocol的摸樣
以EFI_DISKIO_PROTOCOL 來看看Protocol的樣子。 
MdePkg/Include/Protocol/BlockIo.h
:
220
///
///  This protocol provides control over block devices.
///
struct _EFI_BLOCK_IO_PROTOCOL {
  
///
  
/// The revision to which the block IO interface adheres. All future
  
/// revisions must be backwards compatible. If a future version is not
  
/// back wards compatible, it is not the same GUID.
  
///
  UINT64              Revision;
  
///
  
/// Pointer to the EFI_BLOCK_IO_MEDIA data for this device.
  
///
  EFI_BLOCK_IO_MEDIA  *Media;

  EFI_BLOCK_RESET     Reset;
  EFI_BLOCK_READ      ReadBlocks;
  EFI_BLOCK_WRITE     WriteBlocks;
  EFI_BLOCK_FLUSH     FlushBlocks;

};

extern EFI_GUID gEfiBlockIoProtocolGuid;
MdePkg/Include/Protocol/BlockIo.h
:
220
#define EFI_BLOCK_IO_PROTOCOL_GUID \
  { \
    
0x964e5b210x64590x11d2, {0x8e0x390x00xa00xc90x690x720x3b } \
  }

typedef 
struct _EFI_BLOCK_IO_PROTOCOL  EFI_BLOCK_IO_PROTOCOL;
EFI_BLOCK_IO_PROTOCOL 有兩個成員變量,四個成員函數(當然從C的角度來看,“成員函數”叫法不準確,它實際上也是一個成員變量,只是這個變量是函數指針).  gEfiBlockIoProtocolGuid({0x964e5b210x64590x11d2, {0x8e0x390x00xa00xc90x690x720x3b })標示了EFI_BLOCK_IO_PROTOCOL 。
來看成員函數的聲明

/**
  Read BufferSize bytes from Lba into Buffer.

  @param  This       Indicates a pointer to the calling context.
  @param  MediaId    Id of the media, changes every time the media is replaced.
  @param  Lba        The starting Logical Block Address to read from
  @param  BufferSize Size of Buffer, must be a multiple of device block size.
  @param  Buffer     A pointer to the destination buffer for the data. The caller is
                     responsible for either having implicit or explicit ownership of the buffer.

  @retval EFI_SUCCESS           The data was read correctly from the device.
  @retval EFI_DEVICE_ERROR      The device reported an error while performing the read.
  @retval EFI_NO_MEDIA          There is no media in the device.
  @retval EFI_MEDIA_CHANGED     The MediaId does not matched the current device.
  @retval EFI_BAD_BUFFER_SIZE   The Buffer was not a multiple of the block size of the device.
  @retval EFI_INVALID_PARAMETER The read request contains LBAs that are not valid,
                                or the buffer is not on proper alignment.

*
*/
typedef
EFI_STATUS
(EFIAPI 
*EFI_BLOCK_READ)(
  IN EFI_BLOCK_IO_PROTOCOL          
*This,
  IN UINT32                         MediaId,
  IN EFI_LBA                        Lba,
  IN UINTN                          BufferSize,
  OUT VOID                          
*Buffer
  );
EFI_BLOCK_READ具體用法我們先不看,我們來看它的第一個參數,指向EFI_BLOCK_IO_PROTOCOL  對象自己的this指針,這是成員函數區別于一般函數的重要特征。

如何使用Protocol       

使用Protocol之前,我們要弄清楚Protocol位于什么地方。首先我們要來認識一下EFI_HANDLE,
///
/// A collection of related interfaces.
///
typedef VOID                      *EFI_HANDLE;
EFI_HANDLE是指向某種對象的指針,UEFI用它來表示某個對象。 UEFI掃描總線后,會為每個設備建立一個Controller對象,用于控制設備,所有該設備的驅動以protocol的形式安裝到這個controller中,這個Controller就是一個EFI_HANDLE對象。 當我們將一個.efi文件加載到內存中,UEFI也會為該文件建立一個Image對象(此Image非圖像的意識), 這個Image對象也是一個EFI_HANDLE對象。 在UEFI內部,EFI_HANDLE被理解為IHANDLE
///
/// IHANDLE - contains a list of protocol handles
///
typedef struct {
  UINTN               Signature;
  
/// All handles list of IHANDLE
  LIST_ENTRY          AllHandles;
  
/// List of PROTOCOL_INTERFACE's for this handle
  LIST_ENTRY          Protocols;
  UINTN               LocateRequest;
  
/// The Handle Database Key value when this handle was last created or modified
  UINT64              Key;
} IHANDLE;
每個IHANDLE中都有一個Protocols鏈表,存放屬于自己的protocol。所有的IHANDLE通過AllHandles鏈接起來。
要使用Protocol,首先要找到protocol對象,可以通過BootServices的OpenProtocol(...), HandleProtocl(...), LocateProtocol(...)獲得。
typedef
/**
  Queries a handle to determine if it supports a specified protocol. If the protocol is supported by the
  handle, it opens the protocol on behalf of the calling agent.
  @param  Handle                The handle for the protocol interface that is being opened.
  @param  Protocol              The published unique identifier of the protocol.
  @param  Interface             Supplies the address where a pointer to the corresponding Protocol
                                        Interface is returned.
  @param  AgentHandle        The handle of the agent that is opening the protocol interface
                                        specified by Protocol and Interface.
  @param  ControllerHandle    If the agent that is opening a protocol is a driver that follows the
                                        UEFI Driver Model, then this parameter is the controller handle
                                        that requires the protocol interface. If the agent does not follow
                                        the UEFI Driver Model, then this parameter is optional and may
                                        be NULL.
  @param  Attributes            The open mode of the protocol interface specified by Handle
                                        and Protocol.
  @retval EFI_SUCCESS         An item was added to the open list for the protocol interface, and the
                                        protocol interface was returned in Interface.
  @retval EFI_UNSUPPORTED       Handle does not support Protocol.
  @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
  @retval EFI_ACCESS_DENIED     Required attributes can't be supported in current environment.
  @retval EFI_ALREADY_STARTED   Item on the open list already has requierd attributes whose agent
                                                handle is the same as AgentHandle.
**/
EFI_STATUS
(EFIAPI *EFI_OPEN_PROTOCOL)(
  IN  EFI_HANDLE                Handle,
  IN  EFI_GUID                   *Protocol,
  OUT VOID                       **Interface, OPTIONAL
  IN  EFI_HANDLE                AgentHandle,
  IN  EFI_HANDLE                ControllerHandle,
  IN  UINT32                      Attributes
  );
Handle是Protocol的提供者,如果Handle的Protocols鏈表中有該Potocol,Protocol對象的指針寫到*Interface,并返回EFI_SUCCESS;否則 返回EFI_UNSUPPORTED
如果在驅動中調用OpenProtocol(), AgentHandle是擁有該EFI_DRIVER_BINDING_PROTOCOL對象的Handle;ControllerHandle是擁有該驅動的Controller。
如果調用OpenProtocol的是應用程序,那么AgentHandle是該應用對應的Handle,也就main函數的第一個參數。 
ControllerHandle此時可以忽略。
Attributes可以取以下5種值。
#define EFI_OPEN_PROTOCOL_BY_HANDLE_PROTOCOL   0x00000001
#define EFI_OPEN_PROTOCOL_GET_PROTOCOL             0x00000002
#define EFI_OPEN_PROTOCOL_TEST_PROTOCOL           0x00000004
#define EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER  0x00000008
#define EFI_OPEN_PROTOCOL_BY_DRIVER                   0x00000010
#define EFI_OPEN_PROTOCOL_EXCLUSIVE                   0x00000020

HandleProtocol是OpenProtocol的簡化版,因為大部分情況下我們都不需要關心AgentHandle,ControllerHandle和Attributes。
EFI_STATUS
EFIAPI
CoreHandleProtocol (
  IN EFI_HANDLE       UserHandle,
  IN EFI_GUID         
*Protocol,
  OUT VOID            
**Interface
  )
{
  
return CoreOpenProtocol (
          UserHandle,
          Protocol,
          Interface,
          gDxeCoreImageHandle,
          NULL,
          EFI_OPEN_PROTOCOL_BY_HANDLE_PROTOCOL
          );
}
LocateProtocol(...)是從內核中找出指定Protocol的第一個實例。
typedef
EFI_STATUS
LocateProtocol (
IN EFI_GUID 
*Protocol,
IN VOID       
*Registration OPTIONAL,
OUT VOID     
**Interface
);
UEFI內核中某個Protocol的實例可能不止一個,例如每個硬盤及每個分區都有一個EFI_DISK_IO_PROTOCOL實例。LocateProtocol順序搜索HANDLE鏈表,返回找到的第一個該Protocol的實例。
我們可以用BootServices提供的其它函數處理HANDLE和Protocol。
typedef
EFI_STATUS
LocateHandleBuffer (
IN EFI_LOCATE_SEARCH_TYPE SearchType,
IN EFI_GUID                         
*Protocol OPTIONAL,
IN VOID
                              
 
*SearchKey OPTIONAL,
IN OUT UINTN                       
*NoHandles,
OUT EFI_HANDLE                  
**Buffer
);
可以獲得所有支持指定Protocol的HANDLE,SearchType 有三種:AllHandles(查找所有HANDLE), ByRegisterNotify, ByProtocol(查找支持指定Protocol的HANDLE)。NoHandles是找到的HANDLE的數量, Buffer數組由UEFI復雜分配,由用戶負責釋放。
typedef
EFI_STATUS
LocateHandle (
IN EFI_LOCATE_SEARCH_TYPE SearchType,
IN EFI_GUID                        
*Protocol OPTIONAL,
IN VOID                              
*SearchKey OPTIONAL,
IN OUT UINTN                      
*BufferSize,
OUT EFI_HANDLE                  
*Buffer
);
與LocateHandleBuffer相似,只是用戶負責分配和釋放Buffer數組。
typedef
EFI_STATUS
ProtocolsPerHandle (
IN EFI_HANDLE Handle,
OUT EFI_GUID  
***ProtocolBuffer,
OUT UINTN      
*ProtocolBufferCount
);
獲得指定Handle所支持的所有Protocol, UEFI負責分配內存給ProtocolBuffer,用戶負責釋放該內存。

typedef
EFI_STATUS
(EFIAPI 
*EFI_OPEN_PROTOCOL_INFORMATION) (
IN EFI_HANDLE Handle,
IN EFI_GUID    
*Protocol,
OUT EFI_OPEN_PROTOCOL_INFORMATION_ENTRY 
**EntryBuffer,
OUT UINTN     
*EntryCount
);
typedef struct {
  EFI_HANDLE  AgentHandle;
  EFI_HANDLE  ControllerHandle;
  UINT32        Attributes;
  UINT32        OpenCount;
} EFI_OPEN_PROTOCOL_INFORMATION_ENTRY;
OpenProtocolInformation()獲得指定Handle中指定Protocol的打開信息。
SPEC2.3.1第165頁有很好的例子演示了怎么打開一個Protocol, 
EFI_BOOT_SERVICES *gBS;
EFI_HANDLE ImageHandle;
EFI_DRIVER_BINDING_PROTOCOL 
*This;
IN EFI_HANDLE ControllerHandle,
extern EFI_GUID gEfiXyzIoProtocol;
EFI_XYZ_IO_PROTOCOL 
*XyzIo;
EFI_STATUS Status;

Status 
= gBS->OpenProtocol (
ControllerHandle,
&gEfiXyzIoProtocol,
&XyzIo,
ImageHandle,
NULL,
EFI_OPEN_PROTOCOL_BY_HANDLE_PROTOCOL
);

Status 
= gBS->OpenProtocol (
ControllerHandle,
&gEfiXyzIoProtocol,
&XyzIo,
This
->DriverBindingHandle,
ControllerHandle,
EFI_OPEN_PROTOCOL_GET_PROTOCOL
);
打開Protocol之后就可以使用了,最后要通過CloseProtocol關閉打開的Protocol。
typedef
EFI_STATUS
(EFIAPI 
*EFI_CLOSE_PROTOCOL) (
IN EFI_HANDLE Handle,
IN EFI_GUID 
*Protocol,
IN EFI_HANDLE AgentHandle,
IN EFI_HANDLE ControllerHandle
);
通過HandleProtocol和LocateProtocol打開的Protocol因為沒有指定AgentHandle,所以無法關閉。如果一定要去關閉它,要調用OpenProtocolInformation()獲得AgentHandle和ControllerHandle,然后關閉它。
下面看一個完整的例子,用EFI_DISK_IO_PROTOCOL讀取GPT硬盤的分區表
#include <Uefi.h> 
#include <Base.h> 
#include <Library/UefiLib.h>
#include <Library/BaseLib.h>
#include <Library/BaseMemoryLib.h>
#include <Library/UefiBootServicesTableLib.h>
#include <Library/PrintLib.h>
#include <Protocol/DiskIo.h> 
#include <Protocol/BlockIo.h> 
#include <Protocol/DevicePath.h>    
#include <Uefi/UefiGpt.h>
#include <Library/DevicePathLib.h>
EFI_STATUS
EFIAPI
UefiMain(
  IN EFI_HANDLE        ImageHandle,
  IN EFI_SYSTEM_TABLE  *SystemTable
  )
{
        EFI_STATUS                              Status;
        UINTN                                       HandleIndex, HandleCount;
        EFI_HANDLE                               *DiskControllerHandles = NULL;
        EFI_DISK_IO_PROTOCOL               *DiskIo;

        /*找到所有提供 EFI_DISK_IO_PROTOCOL 的Controller  */
        Status = gBS->LocateHandleBuffer(
                        ByProtocol,
                        &gEfiDiskIoProtocolGuid,
                        NULL,
                        &HandleCount,
                        &DiskControllerHandles);

        if (!EFI_ERROR(Status)) {
                CHAR8 gptHeaderBuf[512];
                EFI_PARTITION_TABLE_HEADER* gptHeader = (EFI_PARTITION_TABLE_HEADER*
)gpHeaderBuf;
                for (HandleIndex = 0; HandleIndex < HandleCount; HandleIndex++) {
                       /*打開EFI_DISK_IO_PROTOCOL  */ 
                       Status = gBS->HandleProtocol(
                                        DiskControllerHandles[HandleIndex],
                                        &gEfiDiskIoProtocolGuid,
                                        (VOID**)&DiskIo);

                        if (!EFI_ERROR(Status)){
                                {
                                        EFI_DEVICE_PATH_PROTOCOL                 *DiskDevicePath;
                                        EFI_DEVICE_PATH_TO_TEXT_PROTOCOL   *Device2TextProtocol = 0;
                                        CHAR16*                                             TextDevicePath = 0;
                                          /*1. 打開EFI_DEVICE_PATH_PROTOCOL  */  
                                        Status = gBS->OpenProtocol(
                                                        DiskControllerHandles[HandleIndex],
                                                        &gEfiDevicePathProtocolGuid,
                                                        (VOID**)&DiskDevicePath,
                                                        ImageHandle,
                                                        NULL,
                                                        EFI_OPEN_PROTOCOL_GET_PROTOCOL
                                                        );
                                        if(!EFI_ERROR(Status)){
                                                if(Device2TextProtocol == 0)
                                                        Status = gBS->LocateProtocol(
                                                                        &gEfiDevicePathToTextProtocolGuid,
                                                                        NULL,
                                                                        (VOID**)&Device2TextProtocol
                                                                        );
                                                /*2. 使用 EFI_DEVICE_PATH_PROTOCOL  得到文本格式的Device Path  */  
                                                TextDevicePath = Device2TextProtocol->ConvertDevicePathToText(DiskDevicePath, TRUE, TRUE);
                                                Print(L"%s\n", TextDevicePath);
                                                if(TextDevicePath)gBS->FreePool(TextDevicePath);
                                                /*3. 關閉 EFI_DEVICE_PATH_PROTOCO */   
                                                Status = gBS->CloseProtocol(
                                                                DiskControllerHandles[HandleIndex],
                                                                &gEfiDevicePathProtocolGuid,
                                                                ImageHandle,
                                                                );
                                        }
                                }
                                {
                                        EFI_BLOCK_IO_PROTOCOL* BlockIo = *(EFI_BLOCK_IO_PROTOCOL**) (DiskIo + 1);
                                        EFI_BLOCK_IO_MEDIA* Media = BlockIo->Media;
                                        /*讀1號扇區。  */   
                                        Status = DiskIo->ReadDisk(DiskIo, Media->MediaId, 512, 512, gptHeader);
                                        /*檢查GPT標志。  */    
                                        if((!EFI_ERROR(Status)) &&( gptHeader -> Header.Signature == 0x5452415020494645)){
                                                UINT32 CRCsum;
                                                UINT32 GPTHeaderCRCsum =  (gptHeader->Header.CRC32);
                                                gptHeader->Header.CRC32 = 0;
                                                gBS -> CalculateCrc32(gptHeader , (gptHeader->Header.HeaderSize), &CRCsum);
                                                if(GPTHeaderCRCsum == CRCsum){
                                                // Find out a GPT Header
                                                }

                                        }

                                }

                        }

                }
                gBS->FreePool(DiskControllerHandles);
        }
}
 









Feedback

# re: UEFI 實戰(4) protocol   回復  更多評論   

2012-08-13 13:55 by 劉煒
EFI_BLOCK_IO_PROTOCOL* BlockIo = *(EFI_BLOCK_IO_PROTOCOL**) (DiskIo + 1);

這是說明兩個協議的數據結構連續地放在一起嗎?

# re: UEFI 實戰(4) protocol [未登錄]  回復  更多評論   

2012-08-15 22:02 by djx_zh
@劉煒
是的。
MdeModulePkg\Universal\Disk\DiskIoDxe\DiskIo.h:36
typedef struct {
UINTN Signature;
EFI_DISK_IO_PROTOCOL DiskIo;
EFI_BLOCK_IO_PROTOCOL *BlockIo;
} DISK_IO_PRIVATE_DATA

# re: UEFI 實戰(4) protocol   回復  更多評論   

2012-10-04 16:29 by 關于 gptHeader-> Header.Signature
你好 我接觸 UFEI 不長時間
有個問題
CHAR8 gptHeader[512];
gptHeader-> Header.Signature
這樣寫可以嗎?
gptHeader 應該是個 struct吧
還有能否把inf文件也發上來呢 謝謝了

# re: UEFI 實戰(4) protocol   回復  更多評論   

2012-10-04 17:44 by snowman1101
你好
我復制了你的代碼
編譯后提示 未聲明 DevicePathToText 于是我加入了
#include <Protocol/DevicePathToText.h>
修改所有錯誤以后
又說 外部 gEfiDevicePathToTextProtocolGuid
請問怎么解決呢??梢园l一下完整代碼嗎
謝謝了

# re: UEFI 實戰(4) protocol   回復  更多評論   

2012-10-05 08:37 by djx_zh
代碼中少了一句
EFI_PARTITION_TABLE_HEADER* gptHeader = (EFI_PARTITION_TABLE_HEADER*
)gpHeaderBuf;

在你.inf文件的[Protocols]下面加上
gEfiDevicePathToTextProtocolGuid

# re: UEFI 實戰(4) protocol [未登錄]  回復  更多評論   

2012-10-06 22:06 by snowman1101
多謝指教,已經可以了


# re: UEFI 實戰(4) protocol [未登錄]  回復  更多評論   

2012-11-15 11:47 by ktfun
親,find out a GPT header 后面的內容呢?

# re: UEFI 實戰(4) protocol   回復  更多評論   

2013-01-07 17:43 by linhao
關于:EFI_BLOCK_IO_PROTOCOL* BlockIo = *(EFI_BLOCK_IO_PROTOCOL**) (DiskIo + 1);
我認為這樣更方便理解:
DISK_IO_PRIVATE_DATA *Private_Data;
Private_Data = DISK_IO_PRIVATE_DATA_FROM_THIS (DiskIo);
BlockIo = Private_Data;

# re: UEFI 實戰(4) protocol   回復  更多評論   

2013-01-07 17:45 by linhao
修改:
BlockIo = Private_Data->BlockIo;

# re: UEFI 實戰(4) protocol   回復  更多評論   

2013-01-07 22:12 by djxzh
@linhao
是的。如果你看DiskIO的源碼會發現EDK2就是這樣做的。

# re: UEFI 實戰(4) protocol   回復  更多評論   

2013-02-28 06:44 by howard
hi, 請問如何選擇特定的device進行讀寫?我想parse EFI_DEVICE_PATH_PROTOCOL 的bus id 部分,然后找到目標進行讀寫,但貌似沒有合適的API可以用

Thanks in advance!

# re: UEFI 實戰(4) protocol   回復  更多評論   

2013-03-06 06:08 by djxzh
@howard

EFI_DEVICE_PATH_PROTOCOL 是一個辦法, 確實沒有順手的API, 你要一個node一個node的分析。

也可以用PciIo讀取PCI設備的配置空間,來判斷設備。

# re: UEFI 實戰(4) protocol   回復  更多評論   

2013-03-08 03:29 by howard
恩,我現在暫時用capacity來找我的device. 不知樓主用過EADK嗎,把C標準庫加到EDKII里。 我試了一下,build出來的efi放在shell那一跑就crash with exception, sample efi也一樣問題。。

# re: UEFI 實戰(4) protocol   回復  更多評論   

2014-12-27 03:54 by asam
【UEFI是用C來開發的,C是面向過程的一種語言。而管理和使用UEFI眾多的protocol完全使用面向過程的思想會使程序變得復雜。protocol作為一種對象來設計管理會比較直觀。因而UEFI中的Protocol引入了面向對象的思想,用struct來模擬class, Protocol用struct來實現,用函數指針(Protocol的成員變量)模擬成員函數,此種函數的第一參數必須是指向Protocol的指針,用來模擬this指針。。。。。?!?br>


怪不得,看了hello world,沒有消息機制。特別是和硬件打交道,對這個uefi表示亞歷山大。

# re: UEFI 實戰(4) protocol   回復  更多評論   

2015-05-29 15:07 by 周彬彬
@djx_zh
這個在EDK2 SPE2.4的MdeModulePkg\Universal\Disk\DiskIoDxe\DiskIo.h
這樣定義的:二者并沒有相鄰
#define DISK_IO_PRIVATE_DATA_SIGNATURE SIGNATURE_32 ('d', 's', 'k', 'I')
typedef struct {
UINT32 Signature;

EFI_DISK_IO_PROTOCOL DiskIo;
EFI_DISK_IO2_PROTOCOL DiskIo2;
EFI_BLOCK_IO_PROTOCOL *BlockIo;
EFI_BLOCK_IO2_PROTOCOL *BlockIo2;

UINT8 *SharedWorkingBuffer;

EFI_LOCK TaskQueueLock;
LIST_ENTRY TaskQueue;
} DISK_IO_PRIVATE_DATA;

# re: UEFI 實戰(4) protocol   回復  更多評論   

2015-06-01 04:38 by djx_zh
@周彬彬
對。本文中的方法依賴于 DISK_IO_PRIVATE_DATA的實現,而這個實現沒有一個標準,故不能保證向前兼容。這不是一個安全的方法,只是一個快速的方法。安全的方法還是調用OpenPrototocol或LocateProtocol。

# re: UEFI 實戰(4) protocol   回復  更多評論   

2015-06-02 19:58 by 夏濤
怎么添加關注的啊?

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国产日韩精品在线观看| 最新日韩在线视频| 麻豆精品传媒视频| 亚洲欧美一级二级三级| 亚洲国产精品尤物yw在线观看 | 国产日韩成人精品| 国产手机视频精品| 在线观看欧美亚洲| 日韩亚洲综合在线| 亚洲一区二区三区免费在线观看| 亚洲一区日韩在线| 久久不射网站| 欧美韩日一区二区三区| 一个色综合导航| 久久蜜桃av一区精品变态类天堂| 欧美电影免费观看高清| 国产精品久久久久久久久久免费 | 久久午夜视频| 欧美久久成人| 国产一区二区三区的电影| 亚洲国产成人高清精品| 一区二区三区不卡视频在线观看 | 欧美+亚洲+精品+三区| 亚洲另类在线视频| 欧美一区在线直播| 欧美亚洲成人免费| 亚洲国产精品高清久久久| 亚洲天堂第二页| 美女主播视频一区| 一区二区三区四区在线| 久久久久国产免费免费| 国产精品扒开腿做爽爽爽视频| 黑人一区二区三区四区五区| 一区二区三区久久网| 久久另类ts人妖一区二区| 亚洲免费成人| 久久久夜夜夜| 欧美国产精品中文字幕| 狠狠色综合色综合网络| 亚洲人成免费| 亚洲日韩欧美视频一区| 亚洲一区久久| 午夜精品久久久久| 久久亚洲综合色一区二区三区| 欧美第十八页| 国产视频在线观看一区| 国产精品99久久久久久白浆小说| 欧美成人嫩草网站| 欧美日韩不卡| 一本大道av伊人久久综合| 久久久精品久久久久| 中日韩美女免费视频网址在线观看 | 欧美日本在线观看| 亚洲国产日韩欧美一区二区三区| 欧美中文字幕视频| 亚洲网站在线看| 国产精品久久久久久久久搜平片| 一本一本大道香蕉久在线精品| 欧美国产日产韩国视频| 久久这里只精品最新地址| 精品96久久久久久中文字幕无| 久久久久久一区| 久久精品一区二区三区中文字幕| 国产亚洲欧美日韩日本| 久久精品一二三区| 欧美一区二视频| 国内在线观看一区二区三区| 久久久久国产一区二区| 欧美尤物一区| 亚洲高清久久久| 亚洲福利电影| 欧美屁股在线| 亚洲欧美激情精品一区二区| 99精品国产在热久久婷婷| 欧美日韩在线大尺度| 亚洲一区bb| 欧美亚洲综合久久| 亚洲国产成人精品久久久国产成人一区 | 久久国产视频网| 在线播放国产一区中文字幕剧情欧美 | 国产一区二区三区精品欧美日韩一区二区三区 | 欧美日产国产成人免费图片| 一区二区三区精品在线| 欧美视频1区| 裸体歌舞表演一区二区| 亚洲大片精品永久免费| 亚洲国产精品一区二区三区| 欧美成人综合网站| 亚洲欧美国产精品专区久久| 欧美一区二区成人| 亚洲剧情一区二区| 午夜精品一区二区三区电影天堂 | 日韩香蕉视频| 国产一区二区三区在线免费观看| 免费日韩av片| 国产精品男人爽免费视频1| 美国成人毛片| 国产精品电影网站| 亚洲二区在线| 国产亚洲免费的视频看| 亚洲欧洲午夜| 狠狠入ady亚洲精品经典电影| 亚洲黄一区二区三区| 欧美有码在线观看视频| 最新国产成人在线观看| 亚洲成人中文| 国产精品美女久久久久av超清 | 国产精品卡一卡二卡三| 亚洲国产高清自拍| 国产精品美女久久久久久免费| 尤物九九久久国产精品的特点| 日韩亚洲不卡在线| 久久精品女人| 欧美成人免费视频| 国产精品欧美在线| 美女视频黄a大片欧美| 欧美一区激情| 国产精品美女www爽爽爽视频| 麻豆国产va免费精品高清在线| 欧美日韩亚洲一区二区三区在线观看| 久久在线免费观看视频| 欧美日韩综合在线| 亚洲国产精品女人久久久| 欧美日韩在线第一页| 欧美福利精品| 亚洲精品影视| 亚洲小视频在线观看| 国产麻豆视频精品| 亚洲免费伊人电影在线观看av| 亚洲综合社区| 国产一区二区| 国产精品尤物福利片在线观看| 久久久www| 女人香蕉久久**毛片精品| 久久久久免费| 午夜精品久久久久久久久久久久| 亚洲黄色片网站| 久久国产精品99久久久久久老狼| **性色生活片久久毛片| 欧美日一区二区在线观看| 久久久久久久精| 亚洲美女尤物影院| 欧美激情 亚洲a∨综合| 99pao成人国产永久免费视频| 欧美日本中文字幕| 欧美一区二区三区在线视频| 午夜精品电影| 一区二区三区高清在线观看| 久久精品综合| 亚洲午夜av电影| 91久久国产精品91久久性色| 99国产一区二区三精品乱码| 亚洲一区亚洲| 欧美黄色视屏| 久热国产精品视频| 久久一区二区三区四区五区| 伊人激情综合| 在线不卡a资源高清| 一本一本大道香蕉久在线精品| 亚洲永久网站| 一本久道久久综合中文字幕| 亚洲精品一二三| 久久视频在线视频| 久久成人精品| 欧美mv日韩mv国产网站app| 亚洲一二三区视频在线观看| 欧美成人自拍| 久色婷婷小香蕉久久| 欧美成年人视频网站| 黄色国产精品| 国产精品久久网站| 亚洲激情一区二区| 亚洲欧洲综合另类在线| 欧美成人一区二区三区在线观看 | 日韩一级大片| 久久精品123| 99国产成+人+综合+亚洲欧美| 亚洲欧美激情一区| 亚洲成色777777女色窝| 亚洲一区二区三区久久| 在线观看一区二区视频| 欧美激情va永久在线播放| 亚洲一区二区三区乱码aⅴ| 六月婷婷一区| 亚洲欧美美女| 日韩视频一区二区| 韩日欧美一区二区三区| 欧美日韩成人在线观看| 欧美一区二区视频97| 日韩亚洲欧美一区二区三区| 久久久人成影片一区二区三区| 一区二区三区视频在线观看| 国内精品美女av在线播放| 欧美日韩亚洲综合| 免费在线播放第一区高清av| 午夜天堂精品久久久久 | 久久久夜夜夜| 午夜国产欧美理论在线播放| 亚洲国产老妈| 韩日欧美一区二区|