打印spi GUID

GUID与UUID生成方法

我们经常会遇到需要一些特殊的值来指定一些特殊的变量,而且这个值要绝对的唯一。在windows下被称作GUID,其实GUID是微软对Distributed coumputing environment (DCE) universally unique identifier 的实现,而在Linux下则称作UUID。

不论它叫什么名字,表达的意愿都是一样的,以下用GUID泛指这一概念。它实际上是一个包含128位的bit组。那我们到底如何获取它呢?

先来看看windows平台下如何实现:

通用的GUID的结构如下

typedef struct _GUID {
DWORD Data1;
WORD Data2;
WORD Data3;
BYTE Data4[8];
} GUID;

举一个例子:

假设一个GUID的格式是这样的 6B29FC40-CA47-1067-B31D-00DD010662DA

其中Data1 是32位,可以看做8个四位十六进制数,对应于上面的6B29FC40

其中Data2 是16位,可以看做4个四位十六进制数,对应于上面的CA47

其中Data3 是16位,可以看做4个四位十六进制数,对应于上面的1067

其中Data4 比较特殊,是8个字节也就可以看做16个四位十六进制数

      取其Data4[0],Data4[1]来组成4个四位十六进制数,对应于上面的B31D

      取其Data4[2],Data4[3]来组成4个四位十六进制数,对应于上面的00DD

      取其Data4[4],Data4[5]来组成4个四位十六进制数,对应于上面的0106

      取其Data4[6],Data4[7]来组成4个四位十六进制数,对应于上面的62DA

*注意:四位十六进制数对应一个GUID字符。

在Windows下提供了一个函数可以生成一个GUID。需要使用的头文件 "objbase.h",需要链接的库ole32.lib。

HRESULT CoCreateGuid(GUID * pguid);

在Linux下首先需要下载到相应的库文件和头文件

如果是Debian用户可以方便的通过apt命令来获取相关资源。

     apt-get install uuid-dev

安装完成后会发现

       在/usr/include/ 下有一个uuid文件夹,其中则包含了uuid.h头文件。

       在/usr/lib/ 下有几个libuuid*的链接库文件,包括静态和动态链接库。

打开uuid.h你会发现有一个uuid_generate(uuid_t out) 的函数声明。我们可以通过调用这个函数来生成UUID。

uuid_generate(reinterpret_cast<unsigned char *>(&guid));

生成GUID是不是很简单!是的,因为生成的算法不用我们去实现,我们是站在前人的肩膀所以我们要感谢他们。

下面给出一个WINDOWS和LINUX下通用的程序。

// uuid_test.cpp

#include <string>
#include <stdio.h>
#include <iostream>

#include "uuid_test.hpp"

#ifdef WIN32
#include <objbase.h>
#else
#include <uuid/uuid.h>
#endif

using namespace std;

namespace ChinuxTeam
{
GUID CreateGuid()
{
    GUID guid;
#ifdef WIN32
    CoCreateGuid(&guid);
#else
    uuid_generate(reinterpret_cast<unsigned char *>(&guid));
#endif
    return guid;
}

std::string GuidToString(const GUID &guid)
{
    char buf[64] = {0};
#ifdef __GNUC__
    snprintf(
#else // MSVC
    _snprintf_s(
#endif
                buf,
                sizeof(buf),
                 "{%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}",
                guid.Data1, guid.Data2, guid.Data3,
                guid.Data4[0], guid.Data4[1],
                guid.Data4[2], guid.Data4[3],
                guid.Data4[4], guid.Data4[5],
                guid.Data4[6], guid.Data4[7]);
        return std::string(buf);
}
}

对应的头文件

// uuid_test.hpp"

#include <string>
#include <stdio.h>
#include <iostream>

typedef struct _GUID
{
    unsigned long Data1;
    unsigned short Data2;
    unsigned short Data3;
    unsigned char Data4[8];
} GUID, UUID;

namespace ChinuxTeam

{

GUID CreateGuid();

std::string GuidToString(const GUID &guid);

}// namespace ChinuxTeam

最后一个 GuidToString()稍微解释一下,作用就是把GUID以固定的格式输出到标准string字符串中,

根据编译器的不同调用分别调用了不同的字符串打印函数。

另外附一个简单的测试函数

#include "uuid_test.hpp"
#include <iostream>
#include <string>

using namespace std;
using namespace ChinuxTeam;

int main(int argc, char* argv[])
{
GUID guid_;
guid_ = CreateGuid();
std::string guid_str = GuidToString(guid_);
std::cout << guid_str << std::endl;

return 0;
}

EFI_STATUS SearchByName( VOID ) { ShellPrintEx (-1, -1, L"* SearchByName *\n"); EFI_STATUS Status; EFI_GUID Guid; UINTN DataSize; VOID *Data = NULL; UINT32 Attributes; CHAR16 *Name; CHAR16 *Namestr; CHAR16 *InputStr; UINT64 NameSize, NameBufferSize; CHAR16 *AttributesStr; UINTN EventIndex; EFI_INPUT_KEY key; BOOLEAN Found = FALSE; NameBufferSize = sizeof (CHAR16); Name = AllocateZeroPool (NameBufferSize); Status = ShellPromptForResponse( ShellPromptResponseTypeFreeform, L"Please enter variable name:", (VOID**)&InputStr ); if (EFI_ERROR(Status) || InputStr == NULL || StrLen(InputStr) == 0) { ShellPrintEx(-1, -1, L"Error: Invalid variable name input\n"); FreePool(Name); return EFI_INVALID_PARAMETER; } Namestr = InputStr; /*if (!EFI_ERROR(Status) && InputStr != NULL) { ShellPrintEx(-1, -1, L"You entered: %s\n", InputStr); FreePool(InputStr); }*/ while (TRUE) { NameSize = NameBufferSize; Status = gRT->GetNextVariableName (&NameSize, Name, &Guid); if (Status == EFI_NOT_FOUND) { Status = EFI_SUCCESS; break; } if (Status == EFI_BUFFER_TOO_SMALL) { Name = ReallocatePool (NameBufferSize, NameSize, Name); Status = gRT->GetNextVariableName (&NameSize, Name, &Guid); if (Name == NULL) { return EFI_OUT_OF_RESOURCES; } NameBufferSize = NameSize; } if (StrCmp(Name, Namestr) == 0) { Found = TRUE; DataSize = 0; Status = gRT->GetVariable(Name, &Guid, &Attributes, &DataSize, NULL); if (Status == EFI_BUFFER_TOO_SMALL) { Data = AllocatePool(DataSize); if (Data == NULL) { ShellPrintEx(1, -1, L"Warning: Skipping variable %s (out of memory)\n", Name); continue; } //Detect Variable Status Status = gRT->GetVariable(Name, &Guid, &Attributes, &DataSize, Data); AttributesStr = GetAttrStrType (Attributes); //Output Variable INFO (L"--- --- --- --- ---\n"); ShellPrintEx(-1, -1, L"--- Search Variable INFO ---\n"); ShellPrintEx(-1, -1, L"--- Search Variable Name: %s ---\n", Name); ShellPrintEx(-1, -1, L"--- Search Variable Guid: %g ---\n", &Guid); ShellPrintEx(-1, -1, L"--- Search Variable DataSize: %d ---\n", DataSize); ShellPrintEx(-1, -1, L"--- Search Variable Attributes: %s ---\n", AttributesStr); ShellPrintEx(-1, -1, L"--- --- --- --- ---\n"); FreePool(Data); FreePool(AttributesStr); } } } if (!Found) { ShellPrintEx(-1, -1, L"Error: Variable '%s' not found\n", Namestr); Status = EFI_NOT_FOUND; } if (Namestr) FreePool(Namestr); FreePool(Name); ShellPrintEx (-1, -1, L"Press any key to continue...\n"); gBS->WaitForEvent (1, &gST->ConIn->WaitForKey, &EventIndex); gST->ConIn->ReadKeyStroke (gST->ConIn, &key); gST->ConOut->ClearScreen (gST->ConOut); return EFI_SUCCESS; }详解这段代码的内容
11-20
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值