如何在win98下获取本机的网卡地址(MAC),我用netbios()在win2000下工作正常,但WIN98下却不行(100分)

  • 主题发起人 主题发起人 huhs
  • 开始时间 开始时间
H

huhs

Unregistered / Unconfirmed
GUEST, unregistred user!
如何在win98下获取本机的网卡地址(MAC),我用netbios()在win2000下工作正常,
但WIN98下却不行。
 
应该是本地没有安装netbios协议。
 
下面这个如何?我不知道是否使用Netbios。不过在我的计算机上可以。
uses winsock;

procedure GetComputerNameAndIP;
var
wVersionRequested: WORD;
wsaData: TWSAData;
p: PHostEnt;
s: array[0..128] of char;
p2: pchar;
OutPut: array[0..100] of char;
begin
{Start up WinSock}
wVersionRequested := MAKEWORD(1, 1);
WSAStartup(wVersionRequested, wsaData);

{Get the computer name}
GetHostName(@s, 128);
p := GetHostByName(s);

{Get the IpAddress}
p2 := iNet_ntoa((PInAddr(p^.h_addr_list^))^);
StrPCopy(OutPut, 'Hostname: ' + Format('%s', [p^.h_Name]) + #10#13 +
'IPaddress: ' + Format('%s', [p2]));
WSACleanup;
MessageBox(0, OutPut, 'NetInfo', mb_ok or mb_iconinformation);
end;
 
可以吗?这个没有uses NB30。应该行的。
 
GetComputerNameAndIP 并没有得到网卡地址。
在WIN98下已经安装了netbeui协议。
 
我马上就实验!稍等,记得行的呀。。。
 
可以的呀。
计算机主机名(DNS里面填写的)和IP都显示出来了呀。
我网络安装了
Microsoft 网络客户
Microsoft 友好登陆
Action EN1208 PCI网卡
TCP/IP
Microsoft 网络文件和打印共享。
再就没有其他的。操作系统Windows ME

 
Win98下不行,怎么会?把你的代码帖出来。我用的是Win2000,只能让用98的富翁们
看看了。
 
to Yzhshi 人家要的可不是 IP 阿
我的代码很奇怪只能在 98下获得 Mac 和你的现象正好相反?
 
我的代码如下:
function GetAdapterInfo:string;//(Lana: Char): String;
var
Adapter: TAdapterStatus;
NCB: TNCB;
begin
FillChar(NCB, SizeOf(NCB), 0);
NCB.ncb_command := Char(NCBRESET);
NCB.ncb_lana_num := #0;//Lana;
if Netbios(@NCB) <> Char(NRC_GOODRET) then
begin
Result := '';
Exit;
end;

FillChar(NCB, SizeOf(NCB), 0);
NCB.ncb_command := Char(NCBASTAT);
NCB.ncb_lana_num := #0;//Lana;
NCB.ncb_callname := '*';

FillChar(Adapter, SizeOf(Adapter), 0);
NCB.ncb_buffer := @Adapter;
NCB.ncb_length := SizeOf(Adapter);
if Netbios(@NCB) <> Char(NRC_GOODRET) then
begin
Result := '';
Exit;
end;
Result :=
IntToHex(Byte(Adapter.adapter_address[0]), 2) + '-' +
IntToHex(Byte(Adapter.adapter_address[1]), 2) + '-' +
IntToHex(Byte(Adapter.adapter_address[2]), 2) + '-' +
IntToHex(Byte(Adapter.adapter_address[3]), 2) + '-' +
IntToHex(Byte(Adapter.adapter_address[4]), 2) + '-' +
IntToHex(Byte(Adapter.adapter_address[5]), 2);
end;
在win98和win2000下都安装了NETBEUI协议和TCP/IP协议。但win98下取不到。
我用SendArp()也是在win98下出错,错误原因是win98下不支持该方法。
to itren :能将你的代码贴出来看看吗?

 
// nb.pas
unit nb;

{$F+}
interface


uses SysUtils, Winprocs, Wintypes;

const


{ size of a netbios name }
NBNAMESIZE = 16;

{ max number of network adapters }
{ remeber it's BIG Blue, right ? }
MAXLANAS = 254;

{ NCB Command codes }

NCB_ASYNC = $80; { asynch command bit to be or-ed into command

}

NCB_CALL = $10; { open a session }
NCB_LISTEN = $11; { wait for a call }
NCB_HANGUP = $12; { end session }
NCB_SEND = $14; { send data }
NCB_RECV = $15; { receive data }
NCB_RECVANY = $16; { receive data on any session }
NCB_CHAINSEND = $17; { chain send data }
NCB_DGSEND = $20; { send a datagram }
NCB_DGRECV = $21; { receive datagram }
NCB_DGSENDBC = $22; { send broadcast datagram }
NCB_DGREVCBC = $23; { receive broadcast datagram }
NCB_ADDNAME = $30; { add unique name to local table }
NCB_DELNAME = $31; { delete name from local table }
NCB_RESET = $32; { reset adapter }
NCB_ADPSTAT = $33; { adapter status }
NCB_SSTAT = $34; { session status }
NCB_CANCEL = $35; { cancel NCB request }
NCB_ADDGRPNAME= $36; { add group name to local table }
NCB_ENUM = $37; { enum adapters }
NCB_UNLINK = $70; { unlink remote boot code }
NCB_SENDNA = $71; { send, don't wait for ACK }
NCB_CHAINSENDNA=$72; { chain send, but don't wait for ACK }
NCB_LANSTALERT= $73; { lan status alert }
NCB_ACTION = $77; { enable extensions }
NCB_FINDNAME = $78; { search for name on the network }
NCB_TRACE = $79; { activate / stop tracing }

{ NCB return codes }

NRC_GOODRET = $00; { good return
also returned when ASYNCH request accept

ed }
NRC_BUFLEN = $01; { illegal buffer length

}
NRC_ILLCMD = $03; { illegal command

}
NRC_CMDTMO = $05; { command timed out

}
NRC_INCOMP = $06; { message incomplete, issue another comman

d }
NRC_BADDR = $07; { illegal buffer address

}
NRC_SNUMOUT = $08; { session number out of range

}
NRC_NORES = $09; { no resource available

}
NRC_SCLOSED = $0a; { session closed

}
NRC_CMDCAN = $0b; { command cancelled

}
NRC_DUPNAME = $0d; { duplicate name

}
NRC_NAMTFUL = $0e; { name table full

}
NRC_ACTSES = $0f; { no deletions, name has active sessions

}
NRC_LOCTFUL = $11; { local session table full

}
NRC_REMTFUL = $12; { remote session table full

}
NRC_ILLNN = $13; { illegal name number

}
NRC_NOCALL = $14; { no callname

}
NRC_NOWILD = $15; { cannot put * in NCB_NAME

}
NRC_INUSE = $16; { name in use on remote adapter

}
NRC_NAMERR = $17; { name deleted

}
NRC_SABORT = $18; { session ended abnormally

}
NRC_NAMCONF = $19; { name conflict detected

}
NRC_IFBUSY = $21; { interface busy, IRET before retrying

}
NRC_TOOMANY = $22; { too many commands outstanding, retry lat

er }
NRC_BRIDGE = $23; { ncb_lana_num field invalid

}
NRC_CANOCCR = $24; { command completed while cancel occurring

}
NRC_CANCEL = $26; { command not valid to cancel

}
NRC_DUPENV = $30; { name defined by anther local process

}
NRC_ENVNOTDEF = $34; { environment undefined. RESET required

}
NRC_OSRESNOTAV = $35; { required OS resources exhausted

}
NRC_MAXAPPS = $36; { max number of applications exceeded

}
NRC_NOSAPS = $37; { no saps available for netbios

}
NRC_NORESOURCES = $38; { requested resources are not available

}
NRC_INVADDRESS = $39; { invalid ncb address or length > segment

}
NRC_INVDDID = $3B; { invalid NCB DDID

}
NRC_LOCKFAIL = $3C; { lock of user area failed

}
NRC_OPENERR = $3f; { NETBIOS not loaded

}
NRC_SYSTEM = $40; { system error

}

NRC_PENDING = $ff; { asynchronous command is not yet finished

}

{ Values for transport_id }

ALL_TRANSPORTS = 'M'#$00#$00#$00;
MS_NBF = 'MNBF';


{ values for name_flags bits. }

NAME_FLAGS_MASK = $87;

GROUP_NAME = $80;
UNIQUE_NAME = $00;

REGISTERING = $00;
REGISTERED = $04;
DEREGISTERED = $05;
DUPLICATE = $06;
DUPLICATE_DEREG = $07;


{ Values for state }

LISTEN_OUTSTANDING = $01;
CALL_PENDING = $02;
SESSION_ESTABLISHED = $03;
HANGUP_PENDING = $04;
HANGUP_COMPLETE = $05;
SESSION_ABORTED = $06;


type


{ Netbios Name }
TNBName = array[0..(NBNAMESIZE - 1)] of byte;

{ MAC address }
TMacAddress = array[0..5] of byte;

PNCB = ^TNCB;

{ Netbios Control Block }

{$IFDEF WIN32}
TNCBPostProc = procedure(P: PNCB);
{$ENDIF}

TNCB = packed record { Netbios Control Block }
Command: byte; { command code }
RetCode: byte; { return code }
LSN: byte; { local session number }
Num: byte; { name number }
Buf: ^byte; { data buffer }
Length: word; { data length }
CallName: TNBName; { name to call }
Name: TNBName; { our own name }
RTO: byte; { receive time-out }
STO: byte; { send time-out }
{$IFNDEF WIN32}
Post_Offs:word; { asynch notification routine offset }
Post_Seg: word; { asynch notification routine segment}
{$ELSE}
PostPrc: TNCBPostProc;{ asynch notification routine (nb30) }
{$ENDIF}
Lana_Num: byte; { adapter number }
Cmd_Cplt: byte; { command completion flag }
{$IFDEF WIN32}
Reserved: array[0..9] of byte; { Reserverd for Bios use }
Event: THandle; { WIN32 event handle to be signalled }
{ for asynch cmd completion }
{$ELSE}
Reserved: array[0..13] of byte; { Reserved }
{$ENDIF}
end;


{ Netbios Name Info record }
PNameInfo = ^TNameInfo;
TNameInfo = packed record { name info record }
Name: TNBName; { netbios name }
NameNum:byte; { name number }
NameSt: byte; { name status }
end;

{ Netbios adapter status }
PAdpStat = ^TAdpStat;
TAdpStat = packed record { adapter status record}
ID: TMacAddress; { adapter mac address }
VMajor: byte; { software version major number }
Resvd0: byte;
AdpType: byte; { adapter type }
VMinor: byte; { software version minor number }
RptTime: word; { reporting time period }
RcvCRC: word; { receive crc errors }
RcvOth: word; { receive other errors }
TxmCol: word; { transmit collisions }
TxmOth: word; { transmit other errors }
TxmOK: LongInt; { successfull transmissions }
RcvOK: LongInt; { successfull receives }
TxmRetr: word; { transmit retries }
NoRcvBuf: word; { number of 'no receive buffer' }
T1_tmo: word; { t1 time-outs }
Ti_tmo: word; { ti time_outs }
Resvd1: LongInt;
Free_Ncbs:word; { number of free ncb's }
Cfg_Ncbs: word; { number of configured ncb's }
max_Ncbs: word; { max ncb's used }
NoTxmBuf: word; { number of 'no transmit buffer'}
MaxDGSize:word; { max. datagram size }
Pend_Ses: word; { number of pending sessions }
Cfg_Ses: word; { number of configured sessions }
Max_Ses: word; { max sessions used }
Max_SPSz: word; { max. session packet size }
nNames: word; { number of names in local table}
Names: array[0..15] of TnameInfo; { local name table }
end;

{
Structure returned to the NCB command NCBSSTAT is SESSION_HEADER fo

llowed
by an array of SESSION_BUFFER structures. If the NCB_NAME starts wi

th an
asterisk then an array of these structures is returned containing t

he
status for all names.
}

{ session header }
PSession_Header = ^TSession_Header;
TSession_Header = packed record
sess_name: byte;
num_sess: byte;
rcv_dg_outstanding: byte;
rcv_any_outstanding: byte;
end;

{ session buffer }
PSession_Buffer = ^TSession_Buffer;
TSession_Buffer = packed record
lsn: byte;
state: byte;
local_name: TNBName;
remote_name: TNBName;
rcvs_outstanding: byte;
sends_outstanding: byte;
end;

{
Structure returned to the NCB command NCBENUM.

On a system containing lana's 0, 2 and 3, a structure with
length =3, lana[0]=0, lana[1]=2 and lana[2]=3 will be returned.
}
PLana_Enum = ^TLana_Enum;
TLANA_ENUM = packed record
length: byte; { Number of valid entries in lana[] }
lana: array[0..(MAXLANAS - 1)] of byte;
end;

{
Structure returned to the NCB command NCBFINDNAME is FIND_NAME_HEAD

ER followed
by an array of FIND_NAME_BUFFER structures.
}

PFind_Name_Header = ^TFind_Name_Header;
TFind_Name_Header = packed record
node_count: word;
reserved: byte;
unique_group: byte;
end;

PFind_Name_Buffer = ^TFind_Name_Buffer;
TFind_Name_Buffer = packed record
length: byte;
access_control: byte;
frame_control: byte;
destination_addr:TMacAddress;
source_addr: TMacAddress;
routing_info: array[0..17] of byte;
end;

{
Structure provided with NCBACTION. The purpose of NCBACTION is to p

rovide
transport specific extensions to netbios.
}

PAction_Header = ^TAction_Header;
TAction_Header = packed record
transport_id: LongInt;
action_code: Word;
reserved: Word;
end;



{$IFDEF WIN32}
function Netbios(P: PNCB): Char; stdcall;
{$ENDIF}

{ Exposed functions }


function NetbiosCmd(var NCB: TNCB): Word;


implementation

{$IFDEF WIN32}
function Netbios; external 'netapi32.dll' name 'Netbios';
{$ENDIF}

{---------------------------------}
{ execute a Windows Netbios Call }
{---------------------------------}

function NetbiosCmd(var NCB: TNCB): Word;
begin
{$IFNDEF WIN32}
asm
push bp { save bp }
push ss { save ss }
push ds { save ds }
les bx, NCB { get segment/offset address of NCB }
call NetBiosCall; { 16 bit Windows Netbios call }
xor ah,ah
mov @Result, ax { store return code }
pop ds { restore ds }
pop ss { restore ss }
pop bp { restore bp }
end;
{$ELSE}
Result := Word(Netbios(PNCB(@NCB))); { 32 bit Windows Netbios call }


{$ENDIF}
end;

end.
================================
Function NBGetAdapterAddress(a:Integer) : String;
Var
NCB : TNCB; // Netbios control block //NetBios控制块
ADAPTER : TADAPTERSTATUS; // Netbios adapter status//取网卡状态
LANAENUM : TLANAENUM; // Netbios lana
intIdx : Integer; // Temporary work value//临时变量
cRC : Char; // Netbios return code//NetBios返回值
strTemp : String; // Temporary string//临时变量
Begin
// Initialize
Result := '';
Try
// Zero control blocl
ZeroMemory(@NCB, SizeOf(NCB));
// Issue enum command
NCB.ncb_command := Chr(NCBENUM);
cRC := NetBios(@NCB);
// Reissue enum command
NCB.ncb_buffer := @LANAENUM;
NCB.ncb_length := SizeOf(LANAENUM);
cRC := NetBios(@NCB);
If Ord(cRC)<>0 Then
exit;
// Reset adapter
ZeroMemory(@NCB, SizeOf(NCB));
NCB.ncb_command := Chr(NCBRESET);
NCB.ncb_lana_num := LANAENUM.lana[a];
cRC := NetBios(@NCB);
If Ord(cRC)<>0 Then
exit;
// Get adapter address
ZeroMemory(@NCB, SizeOf(NCB));
NCB.ncb_command := Chr(NCBASTAT);
NCB.ncb_lana_num := LANAENUM.lana[a];
StrPCopy(NCB.ncb_callname, '*');
NCB.ncb_buffer := @ADAPTER;
NCB.ncb_length := SizeOf(ADAPTER);
cRC := NetBios(@NCB);
// Convert it to string
strTemp := '';
For intIdx := 0 To 5 Do
strTemp := strTemp + InttoHex(Integer(ADAPTER.adapter_address[in
tIdx]),2);
Result := strTemp;
Finally
End;
End;

//Main.pas
unit main;

interface

uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
StdCtrls;

type
TForm1 = class(TForm)
Button1: TButton;
Memo1: TMemo;
procedure FormCreate(Sender: TObject);
procedure Button1Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;

var
Form1: TForm1;

implementation

uses nb;

{$R *.DFM}
{---------------------------------------------}
{ enumerate the lana's - works only on WIN32 }
{---------------------------------------------}

function NbLanaEnum: TLana_Enum;
var
NCB: TNCB;
L_Enum: TLana_Enum;
RetCode: Word;
begin
{$IFDEF WIN32}
FillChar(NCB, SizeOf(NCB), 0);
FillChar(L_Enum, SizeOf(TLana_Enum), 0);
NCB.Command := NCB_ENUM;
NCB.Buf := @L_Enum;
NCB.Length := Sizeof(L_Enum);
RetCode := NetBiosCmd(NCB);
if RetCode <> NRC_GOODRET then begin
L_Enum.Length := 0;
L_Enum.Lana[0] := Byte(RetCode);
end;
{$ELSE} { not supported for WIN16, fake LANA 0 }

L_Enum.Length := 1;
L_Enum.Lana[0] := 0;
{$ENDIF}
Result := L_Enum;
end;

{----------------------------------------}
{ Reset the lana - don't for WIN16 ! }
{----------------------------------------}
function NbReset(l: Byte): Word;
var
NCB: TNCB;
begin
{$IFNDEF WIN32} { will reset all your connections for WIN1

6 }
Result := NRC_GOODRET; { so just fake a reset for Win16

}
{$ELSE}
FillChar(NCB, SizeOf(NCB), 0);
NCB.Command := NCB_RESET;
NCB.Lana_Num := l;
Result := NetBiosCmd(NCB);
{$ENDIF}
end;
{----------------------------------------}
{ return the MAC address of an interface }
{ in the form of a string like : }
{ 'xx:xx:xx:xx:xx:xx' }
{ using the definitions in nb.pas }
{----------------------------------------}

function NbGetMacAddr(LanaNum: Integer): String;
var
NCB: TNCB;
AdpStat: TAdpStat;
RetCode: Word;
begin
FillChar(NCB, SizeOf(NCB), 0);
FillChar(AdpStat, SizeOf(AdpStat), 0);
NCB.Command := NCB_ADPSTAT;
NCB.Buf := @AdpStat;
NCB.Length := Sizeof(AdpStat);
FillChar(NCB.CallName, Sizeof(TNBName), $20);
NCB.CallName[0] := Byte('*');
NCB.Lana_Num := LanaNum;
RetCode := NetBiosCmd(NCB);
if RetCode = NRC_GOODRET then begin
Result := Format('%2.2x:%2.2x:%2.2x:%2.2x:%2.2x:%2.2x',
[AdpStat.ID[0],
AdpStat.ID[1],
AdpStat.ID[2],
AdpStat.ID[3],
AdpStat.ID[4],
AdpStat.ID[5]
]);
end else begin
Result := '??:??:??:??:??:??';
end;
end;


procedure TForm1.Button1Click(Sender: TObject);
begin
Close;
end;

procedure TForm1.FormCreate(Sender: TObject);
var
L_Enum : TLana_Enum;
RetCode: Word;
i: Integer;
begin
L_Enum := NbLanaEnum; { enumerate lanas for WIN NT }
if L_Enum.Length = 0 then begin
Button1.Caption := Format('LanaEnum err=%2.2x', [L_Enum.Lana[0]]);
exit;
end;

for i := 0 to (L_Enum.Length - 1)do begin { for every lana found

}

RetCode := NbReset(L_Enum.Lana); { Reset lana for WIN NT

}
if RetCode <> NRC_GOODRET then begin
Button1.Caption := Format('Reset Lana %d err=%2.2x',[i, RetCode

]);
exit;
end;
{ Get MAC Address

}
Memo1.Lines.Add(Format('Lana %x = %s', [L_Enum.Lana, NbGetMacAddr(i)]));

end;

Button1.Caption := 'Stop';
end;



end.

//DemoGetmac.dpr

program DemoGetMac;

uses
Forms,
main in 'main.pas' {Form1},
nb in 'nb.pas';

{$R *.RES}

begin
Application.Initialize;
Application.CreateForm(TForm1, Form1);
Application.Run;
end.




 
问题还是在于需要安装netbios协议。
 

Similar threads

D
回复
0
查看
1K
DelphiTeacher的专栏
D
D
回复
0
查看
890
DelphiTeacher的专栏
D
D
回复
0
查看
1K
DelphiTeacher的专栏
D
D
回复
0
查看
1K
DelphiTeacher的专栏
D
D
回复
0
查看
949
DelphiTeacher的专栏
D
后退
顶部