VC获取硬盘序列号 网卡 cpuid bios

来源:互联网 发布:淘宝商家客服几点上班 编辑:程序博客网 时间:2024/05/17 05:13

用api函数读取硬盘的序列号

关于盘序列号有两种:     
    
  硬盘序列号:   英文名   Hard   Disk   Serial   Number,   该号是出厂时生产厂家为     
  区别产品而设置的,   是唯一的,   是只读的,   利用硬盘序列号的     
  加密往往是利用其唯一和只读的特性,   大多是针对有序列号的     
  IDE   HDD而言,   对于没有序列号或SCSI   HDD硬盘则无能为力,     
  这也是利用它进行加密的局限性.     
  卷的序列号:   英文名   Volume   Serial   Number,   该号既可指软磁盘要得,   如:     
  A:盘和B:盘的,   又可以指硬盘的逻辑盘,   如:   C:,   D:...的,     
  是高级格式化时随机产生的,   是可以修改的,   所以利用其进行     
  加密,  

在写程序时我们想对每一台计算机都生成一个唯一的标识,而且在一此共享软件中我们也看到,软件在不同的机器上生成了不同的标识,这是如何实现的呢,其实是软件有一部分程序读取了,本地计算机的一部分硬件参数(如硬盘序列号,网卡序列号等等),再通过一系列算法而得到了一个唯一标识,其实我们也可以通过一个api函数生成我们的唯一标识,由于硬盘人人都有,而网卡不一定每个人都有,所以以读硬盘序列号为例,
下面就先介绍一下我们要用到的api函数
 BOOL GetVolumeInformation(
  LPCTSTR lpRootPathName,           // 硬盘的路径
  LPTSTR lpVolumeNameBuffer,        // 硬盘的卷标
  DWORD nVolumeNameSize,            // 卷标的字符串长度
  LPDWORD lpVolumeSerialNumber,     // 硬盘的序列号
  LPDWORD lpMaximumComponentLength, // 最大的文件长度
  LPDWORD lpFileSystemFlags,        // 文件系统的一此标志
  LPTSTR lpFileSystemNameBuffer,    // 存储所在盘符的分区类型的长指针变量
  DWORD nFileSystemNameSize         // 分区类型的长指针变量所指向的字符串长度
);
如果上述函数成功就返回一个非0值。

光说不做,是不行了,我们还得实践一下:// 最大的文件长度
首选用MFC AppWizard建立一个基于Dialog base的对话框工程,名为GetHardID,点击finish。
加一个按钮,双击它,点击ok,并在对应的函数中加入如下代码:
 LPCTSTR lpRootPathName="c://"; //取C盘
 LPTSTR lpVolumeNameBuffer=new char[12];//磁盘卷标
 DWORD nVolumeNameSize=12;// 卷标的字符串长度
 DWORD VolumeSerialNumber;//硬盘序列号
 DWORD MaximumComponentLength;// 最大的文件长度
 LPTSTR lpFileSystemNameBuffer=new char[10];// 存储所在盘符的分区类型的长指针变量
 DWORD nFileSystemNameSize=10;// 分区类型的长指针变量所指向的字符串长度
 DWORD FileSystemFlags;// 文件系统的一此标志
 ::GetVolumeInformation(lpRootPathName,
  lpVolumeNameBuffer, nVolumeNameSize,
  &VolumeSerialNumber, &MaximumComponentLength,
  &FileSystemFlags,
  lpFileSystemNameBuffer, nFileSystemNameSize);
 
 CString str;
 str.Format("Seria Num is %lx ",VolumeSerialNumber);
 AfxMessageBox(str);
 
编译,链接并运行程序,单击按钮,在弹出的对话框中就是我们要的序列号。
对不对我们验证一下,进入dos窗口,打入"dir c:/p"命令,怎么样是不是和我们的程序显示的一模一样
这样我们就在功告成了。

=======cpu adapter id ===========

以下代码可以取得系统特征码(网卡MAC、硬盘序列号、CPU ID、BIOS编号)

    BYTE szSystemInfo[4096]; // 在程序执行完毕后,此处存储取得的系统特征码
    UINT uSystemInfoLen = 0; // 在程序执行完毕后,此处存储取得的系统特征码的长度

    // 网卡 MAC 地址,注意: MAC 地址是可以在注册表中修改的
    {
        UINT uErrorCode = 0;
        IP_ADAPTER_INFO iai;
        ULONG uSize = 0;
        DWORD dwResult = GetAdaptersInfo( &iai, &uSize );
        if( dwResult == ERROR_BUFFER_OVERFLOW )
        {
            IP_ADAPTER_INFO* piai = ( IP_ADAPTER_INFO* )HeapAlloc( GetProcessHeap( ), 0, uSize );
            if( piai != NULL )
            {
                dwResult = GetAdaptersInfo( piai, &uSize );
                if( ERROR_SUCCESS == dwResult )
                {
                    IP_ADAPTER_INFO* piai2 = piai;
                    while( piai2 != NULL && ( uSystemInfoLen + piai2->AddressLength ) < 4096U )
                    {
                        CopyMemory( szSystemInfo + uSystemInfoLen, piai2->Address, piai2->AddressLength );
                        uSystemInfoLen += piai2->AddressLength;
                        piai2 = piai2->Next;                        
                    }
                }
                else
                {
                    uErrorCode = 0xF0000000U + dwResult;
                }
                VERIFY( HeapFree( GetProcessHeap( ), 0, piai ) );
            }
            else
            {
                return FALSE;
            }
        }
        else
        {
            uErrorCode = 0xE0000000U + dwResult;
        }
        if( uErrorCode != 0U )
        {
            return FALSE;
        }
    }

    // 硬盘序列号,注意:有的硬盘没有序列号
    {
        OSVERSIONINFO ovi = { 0 };
        ovi.dwOSVersionInfoSize = sizeof( OSVERSIONINFO );
        GetVersionEx( &ovi );
        
        if( ovi.dwPlatformId != VER_PLATFORM_WIN32_NT )
        {
            // Only Windows 2000, Windows XP, Windows Server 2003...
            return FALSE;
        }
        else
        {
            if( !WinNTHDSerialNumASPhysicalRead( szSystemInfo, &uSystemInfoLen, 1024 ) )
            {
                WinNTHDSerialNumAsScsiRead( szSystemInfo, &uSystemInfoLen, 1024 );
            }
        }
    }

    // CPU ID
    {
        BOOL bException = FALSE;
        BYTE szCpu[16]  = { 0 };
        UINT uCpuID     = 0U;

        __try 
        {
            _asm 
            {
                mov eax, 0
                cpuid
                mov dword ptr szCpu[0], ebx
                mov dword ptr szCpu[4], edx
                mov dword ptr szCpu[8], ecx
                mov eax, 1
                cpuid
                mov uCpuID, edx
            }
        }
        __except( EXCEPTION_EXECUTE_HANDLER )
        {
            bException = TRUE;
        }
        
        if( !bException )
        {
            CopyMemory( szSystemInfo + uSystemInfoLen, &uCpuID, sizeof( UINT ) );
            uSystemInfoLen += sizeof( UINT );

            uCpuID = strlen( ( char* )szCpu );
            CopyMemory( szSystemInfo + uSystemInfoLen, szCpu, uCpuID );
            uSystemInfoLen += uCpuID;
        }
    }
    
    // BIOS 编号,支持 AMI, AWARD, PHOENIX
    {
        SIZE_T ssize;

        LARGE_INTEGER so; 
        so.LowPart=0x000f0000;
        so.HighPart=0x00000000; 
        ssize=0xffff; 
        wchar_t strPH[30]=L//device//physicalmemory; 

        DWORD ba=0;

        UNICODE_STRING struniph; 
        struniph.Buffer=strPH; 
        struniph.Length=0x2c; 
        struniph.MaximUMLength =0x2e; 

        OBJECT_ATTRIBUTES obj_ar; 
        obj_ar.Attributes =64;
        obj_ar.Length =24;
        obj_ar.ObjectName=&struniph;
        obj_ar.RootDirectory=0; 
        obj_ar.SecurityDescriptor=0; 
        obj_ar.SecurityQualityOfService =0; 

        HMODULE hinstLib = LoadLibrary("ntdll.dll"); 
        ZWOS ZWopenS=(ZWOS)GetProcAddress(hinstLib,"ZwOpenSection"); 
        ZWMV ZWmapV=(ZWMV)GetProcAddress(hinstLib,"ZwMapViewOfSection"); 
        ZWUMV ZWunmapV=(ZWUMV)GetProcAddress(hinstLib,"ZwUnmapViewOfSection"); 
        
        //调用函数,对物理内存进行映射 
        HANDLE hSection; 
        if( 0 == ZWopenS(&hSection,4,&obj_ar) && 
            0 == ZWmapV( 
            ( HANDLE )hSection,   //打开Section时得到的句柄 
            ( HANDLE )0xFFFFFFFF, //将要映射进程的句柄, 
            &ba,                  //映射的基址 
            0,
            0xFFFF,               //分配的大小 
            &so,                  //物理内存的地址 
            &ssize,               //指向读取内存块大小的指针 
            1,                    //子进程的可继承性设定 
            0,                    //分配类型 
            2                     //保护类型 
            ) )
        //执行后会在当前进程的空间开辟一段64k的空间,并把f000:0000到f000:ffff处的内容映射到这里 
        //映射的基址由ba返回,如果映射不再有用,应该用ZwUnmapViewOfSection断开映射 
        {
            BYTE* PBiosSerial = ( BYTE* )ba;
            UINT uBiosSerialLen = FindAwardBios( &pBiosSerial );
            if( uBiosSerialLen == 0U )
            {
                uBiosSerialLen = FindAmiBios( &pBiosSerial );
                if( uBiosSerialLen == 0U )
                {
                    uBiosSerialLen = FindPhoenixBios( &pBiosSerial );
                }
            }
            if( uBiosSerialLen != 0U )
            {
                CopyMemory( szSystemInfo + uSystemInfoLen, pBiosSerial, uBiosSerialLen );
                uSystemInfoLen += uBiosSerialLen;
            }
            ZWunmapV( ( HANDLE )0xFFFFFFFF, ( void* )ba );
        }
    }
    // 完毕, 系统特征码已取得。

原创粉丝点击