请教一下取硬盘序列号的这个函数

pbsql 2004-08-10 07:47:23
下面这个函数相信很多人都看过:
function GetIdeDiskSerialNumber : String;
type
TSrbIoControl = packed record
HeaderLength : ULONG;
Signature : Array[0..7] of Char;
Timeout : ULONG;
ControlCode : ULONG;
ReturnCode : ULONG;
Length : ULONG;
end;
SRB_IO_CONTROL = TSrbIoControl;
PSrbIoControl = ^TSrbIoControl;

TIDERegs = packed record
bFeaturesReg : Byte; // Used for specifying SMART "commands".
bSectorCountReg : Byte; // IDE sector count register
bSectorNumberReg : Byte; // IDE sector number register
bCylLowReg : Byte; // IDE low order cylinder value
bCylHighReg : Byte; // IDE high order cylinder value
bDriveHeadReg : Byte; // IDE drive/head register
bCommandReg : Byte; // Actual IDE command.
bReserved : Byte; // reserved. Must be zero.
end;
IDEREGS = TIDERegs;
PIDERegs = ^TIDERegs;

TSendCmdInParams = packed record
cBufferSize : DWORD;
irDriveRegs : TIDERegs;
bDriveNumber : Byte;
bReserved : Array[0..2] of Byte;
dwReserved : Array[0..3] of DWORD;
bBuffer : Array[0..0] of Byte;
end;
SENDCMDINPARAMS = TSendCmdInParams;
PSendCmdInParams = ^TSendCmdInParams;

TIdSector = packed record
wGenConfig : Word;
wNumCyls : Word;
wReserved : Word;
wNumHeads : Word;
wBytesPerTrack : Word;
wBytesPerSector : Word;
wSectorsPerTrack : Word;
wVendorUnique : Array[0..2] of Word;
sSerialNumber : Array[0..19] of Char;
wBufferType : Word;
wBufferSize : Word;
wECCSize : Word;
sFirmwareRev : Array[0..7] of Char;
sModelNumber : Array[0..39] of Char;
wMoreVendorUnique : Word;
wDoubleWordIO : Word;
wCapabilities : Word;
wReserved1 : Word;
wPIOTiming : Word;
wDMATiming : Word;
wBS : Word;
wNumCurrentCyls : Word;
wNumCurrentHeads : Word;
wNumCurrentSectorsPerTrack : Word;
ulCurrentSectorCapacity : ULONG;
wMultSectorStuff : Word;
ulTotalAddressableSectors : ULONG;
wSingleWordDMA : Word;
wMultiWordDMA : Word;
bReserved : Array[0..127] of Byte;
end;
PIdSector = ^TIdSector;

const
IDE_ID_FUNCTION = $EC;
IDENTIFY_BUFFER_SIZE = 512;
DFP_RECEIVE_DRIVE_DATA = $0007c088;
IOCTL_SCSI_MINIPORT = $0004d008;
IOCTL_SCSI_MINIPORT_IDENTIFY = $001b0501;
DataSize = sizeof(TSendCmdInParams)-1+IDENTIFY_BUFFER_SIZE;
BufferSize = SizeOf(SRB_IO_CONTROL)+DataSize;
W9xBufferSize = IDENTIFY_BUFFER_SIZE+16;
var
hDevice : THandle;
cbBytesReturned : DWORD;
pInData : PSendCmdInParams;
pOutData : Pointer; // PSendCmdOutParams
Buffer : Array[0..BufferSize-1] of Byte;
srbControl : TSrbIoControl absolute Buffer;

procedure ChangeByteOrder( var Data; Size : Integer );
var ptr : PChar;
i : Integer;
c : Char;
begin
ptr := @Data;
for i := 0 to (Size shr 1)-1 do
begin
c := ptr^;
ptr^ := (ptr+1)^;
(ptr+1)^ := c;
Inc(ptr,2);
end;
end;

begin
Result := '';
FillChar(Buffer,BufferSize,#0);
if Win32Platform=VER_PLATFORM_WIN32_NT then
begin // Windows NT, Windows 2000
// Get SCSI port handle
hDevice := CreateFile( '\\.\Scsi0:',
GENERIC_READ or GENERIC_WRITE,
FILE_SHARE_READ or FILE_SHARE_WRITE,
nil, OPEN_EXISTING, 0, 0 );
if hDevice=INVALID_HANDLE_VALUE then Exit;
try
srbControl.HeaderLength := SizeOf(SRB_IO_CONTROL);
System.Move('SCSIDISK',srbControl.Signature,8);
srbControl.Timeout := 2;
srbControl.Length := DataSize;
srbControl.ControlCode := IOCTL_SCSI_MINIPORT_IDENTIFY;
pInData := PSendCmdInParams(PChar(@Buffer)
+SizeOf(SRB_IO_CONTROL));
pOutData := pInData;
with pInData^ do
begin
cBufferSize := IDENTIFY_BUFFER_SIZE;
bDriveNumber := 0;
with irDriveRegs do
begin
bFeaturesReg := 0;
bSectorCountReg := 1;
bSectorNumberReg := 1;
bCylLowReg := 0;
bCylHighReg := 0;
bDriveHeadReg := $A0;
bCommandReg := IDE_ID_FUNCTION;
end;
end;
if not DeviceIoControl( hDevice, IOCTL_SCSI_MINIPORT,
@Buffer, BufferSize, @Buffer, BufferSize,
cbBytesReturned, nil ) then Exit;
finally
CloseHandle(hDevice);
end;
end
else
begin // Windows 95 OSR2, Windows 98
hDevice := CreateFile( '\\.\SMARTVSD', 0, 0, nil,
CREATE_NEW, 0, 0 );
if hDevice=INVALID_HANDLE_VALUE then Exit;
try
pInData := PSendCmdInParams(@Buffer);
pOutData := @pInData^.bBuffer;
with pInData^ do
begin
cBufferSize := IDENTIFY_BUFFER_SIZE;
bDriveNumber := 0;
with irDriveRegs do
begin
bFeaturesReg := 0;
bSectorCountReg := 1;
bSectorNumberReg := 1;
bCylLowReg := 0;
bCylHighReg := 0;
bDriveHeadReg := $A0;
bCommandReg := IDE_ID_FUNCTION;
end;
end;
if not DeviceIoControl( hDevice, DFP_RECEIVE_DRIVE_DATA,
pInData, SizeOf(TSendCmdInParams)-1, pOutData,
W9xBufferSize, cbBytesReturned, nil ) then Exit;
finally
CloseHandle(hDevice);
end;
end;
with PIdSector(PChar(pOutData)+16)^ do
begin
ChangeByteOrder(sSerialNumber,SizeOf(sSerialNumber));
SetString(Result,sSerialNumber,SizeOf(sSerialNumber));
end;
end;
...全文
696 45 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
45 条回复
切换为时间正序
请发表友善的回复…
发表回复
ghost5 2005-04-04
  • 打赏
  • 举报
回复
收藏
wdqxxx 2004-09-10
  • 打赏
  • 举报
回复
太难了,
yuanyu368 2004-08-31
  • 打赏
  • 举报
回复
收藏一下
ztenv 2004-08-31
  • 打赏
  • 举报
回复
关注一下
pbsql 2004-08-31
  • 打赏
  • 举报
回复
问题多多,放弃了
deepWATERblue 2004-08-23
  • 打赏
  • 举报
回复
看一看呀,看一看呀。
smilelhh 2004-08-23
  • 打赏
  • 举报
回复
关注一下,反正我是取硬盘系统号的,就用你那个函数,没深究。
pbsql 2004-08-23
  • 打赏
  • 举报
回复
哎,每种方法都有缺陷,我是不是应该放弃了
pbsql 2004-08-13
  • 打赏
  • 举报
回复
Dlwxn(流云):这倒是个好建议,但CPU序列号、主板序列号有没有通用的获取方法呢?
Dlwxn 2004-08-13
  • 打赏
  • 举报
回复
根据这种情况,我的做法是:硬盘序列号+CPU序列号+主板序列号,三种组合到一起的都没有序列号

的情况应该很少,如果某一个没有序列号,就给一个默认的值。
pbsql 2004-08-13
  • 打赏
  • 举报
回复
看来是问题多多:
1.取硬盘序列号:有的硬盘没有序列号(如三星),且没有在各种操作系统(98/2K/XP)下的通用的获取方法
2.CPU序列号:有的CPU就没有序号
3.主板序列号:有的主板也没有序号
4.网卡序列号:很多机器不装网卡,也不行

哎,不知道计算机里那个部件的序列号是唯一的,并且有通用的获取方法?(用于软件注册)
bluesky23 2004-08-12
  • 打赏
  • 举报
回复
偶也用这个函数。
不过没深入研究过。
呵呵
UP
hhzh426 2004-08-12
  • 打赏
  • 举报
回复
研究一下DeviceIoControl函数,看看是不是能够找到解决办法
还有,如果cmos中设置不用S.M.A.R.T控制硬盘,则smartvsd就不能管理该硬盘。
pbsql 2004-08-11
  • 打赏
  • 举报
回复
那用什么呢?
xiaoxiao197821 2004-08-11
  • 打赏
  • 举报
回复
我已经放弃用硬盘ID号了
ly_liuyang 2004-08-11
  • 打赏
  • 举报
回复
呵呵
我公司的项目都放弃Win98用户了:)
drunkboy 2004-08-11
  • 打赏
  • 举报
回复
UP
pandarus 2004-08-11
  • 打赏
  • 举报
回复
同意楼上
pbsql 2004-08-11
  • 打赏
  • 举报
回复
crossbow(【带三尺剑立不世之功】):那用什么方法呢?
crossbow 2004-08-11
  • 打赏
  • 举报
回复
最好不要用这种硬件识别法,现在每个人都有N个硬盘。
加载更多回复(25)

5,927

社区成员

发帖
与我相关
我的任务
社区描述
Delphi 开发及应用
社区管理员
  • VCL组件开发及应用社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧