uses\r
dnscore,\r
binipstuff,\r
- {$ifdef win32}\r
+ {$ifdef mswindows}\r
winsock,\r
windows,\r
{$else}\r
\r
//convert a name to an IP\r
//will return v4 or v6 depending on what seems favorable, or manual preference setting\r
-//on error the binip will have a family of 0 (other fiels are also currently\r
+//on error the binip will have a family of 0 (other fields are also currently\r
//zeroed out but may be used for further error information in future)\r
-//timeout is in miliseconds, it is ignored when using windows dns\r
+//timeout is in milliseconds, it is ignored when using windows dns\r
function forwardlookup(name:ansistring;timeout:integer):tbinip;\r
\r
//convert a name to a list of all IP's returned\r
\r
toport='53';\r
\r
-var\r
- id:integer;\r
-\r
- sendquerytime:array[0..numsock-1] of integer;\r
implementation\r
\r
-{$ifdef win32}\r
+{$ifdef mswindows}\r
uses dnswin;\r
{$endif}\r
\r
\r
-{$ifndef win32}\r
+{$ifndef mswindows}\r
{$define syncdnscore}\r
{$endif}\r
\r
{$i unixstuff.inc}\r
\r
+type tdnsstatearr=array[0..numsock-1] of tdnsstate;\r
\r
{$ifdef syncdnscore}\r
-var\r
- numsockused:integer;\r
- fd:array[0..numsock-1] of integer;\r
- state:array[0..numsock-1] of tdnsstate;\r
- toaddr:array[0..numsock-1] of tbinip;\r
\r
-{$ifdef win32}\r
+\r
+{$ifdef mswindows}\r
const\r
winsocket = 'wsock32.dll';\r
function sendto(s: TSocket; const Buf; len, flags: Integer; var addrto: TinetSockAddrV; tolen: Integer): Integer; stdcall; external winsocket name 'sendto';\r
\r
\r
function getts:integer;\r
-{$ifdef win32}\r
+{$ifdef mswindows}\r
begin\r
result := GetTickCount and tsmask;\r
{$else}\r
var\r
temp:ttimeval;\r
begin\r
- gettimeofday(temp);\r
+ gettimemonotonic(temp);\r
result := ((temp.tv_usec div 1000) + (temp.tv_sec * 1000)) and tsmask;\r
{$endif}\r
end;\r
\r
-\r
-function sendquery(socknum:integer;const packet:tdnspacket;len:integer):boolean;\r
+procedure resolveloop(timeout:integer;var state:tdnsstatearr;numsockused:integer);\r
var\r
- ip : tbinip;\r
- port : ansistring;\r
- inaddr : TInetSockAddrV;\r
-begin\r
-{ writeln('sendquery ',decodename(state.packet,state.packetlen,12,0,a),' ',state.requesttype);}\r
- result := false;\r
- if len = 0 then exit; {no packet}\r
+ selectresult : integer;\r
+ fds : fdset;\r
\r
- ip := getcurrentsystemnameserverbin(id);\r
+ endtime : longint;\r
+ starttime : longint;\r
+ wrapmode : boolean;\r
+ currenttime : integer;\r
\r
- {$ifdef ipv6}{$ifdef win32}\r
- if toaddr[socknum].family = AF_INET6 then if (useaf = 0) then useaf := useaf_preferv6;\r
- {$endif}{$endif}\r
+ lag : ttimeval;\r
+ selecttimeout : ttimeval;\r
+ socknum:integer;\r
+ needprocessing:array[0..numsock-1] of boolean;\r
+ finished:array[0..numsock-1] of boolean;\r
+ a,b:integer;\r
\r
- port := toport;\r
- toaddr[socknum] := ip;\r
- makeinaddrv(toaddr[socknum],port,inaddr);\r
+ Src : TInetSockAddrV;\r
+ Srcx : {$ifdef mswindows}sockaddr_in{$else}TInetSockAddrV{$endif} absolute Src;\r
+ SrcLen : Integer;\r
+ fromip:tbinip;\r
+ fromport:ansistring;\r
+\r
+ fd:array[0..numsock-1] of integer;\r
+ toaddr:array[0..numsock-1] of tbinip;\r
+ id:integer;\r
+ sendquerytime:array[0..numsock-1] of integer;\r
\r
- sendto(fd[socknum],packet,len,0,inaddr,inaddrsize(inaddr));\r
- sendquerytime[socknum] := getts;\r
- result := true;\r
-end;\r
\r
procedure setupsocket;\r
var\r
inAddrtemp : TInetSockAddrV;\r
- a:integer;\r
biniptemp:tbinip;\r
-\r
+ a,retrycount,porttemp:integer;\r
+ bindresult:boolean;\r
begin\r
- //init both sockets smultaneously, always, so they get succesive fd's\r
-\r
- {recreate sockets every time, reusing them will fail (hang) if the nameserver is changed\r
- also changing the nameserver can't possibly work}\r
- {if fd[0] > 0 then exit;}\r
- for a := 0 to numsock-1 do if (fd[a] > 0) then closesocket(fd[a]);\r
-\r
biniptemp := getcurrentsystemnameserverbin(id);\r
//must get the DNS server here so we know to init v4 or v6\r
\r
\r
\r
for a := 0 to numsockused-1 do begin\r
- makeinaddrv(biniptemp,inttostr( 1024 + randominteger(65536 - 1024) ),inaddrtemp);\r
+ retrycount := 5;\r
+ repeat\r
+ if (retrycount <= 1) then begin\r
+ porttemp := 0; //for the last attempt let the OS decide\r
+ end else begin\r
+ porttemp := 1024 + randominteger(65536 - 1024);\r
+ end;\r
\r
- fd[a] := Socket(biniptemp.family,SOCK_DGRAM,0);\r
+ makeinaddrv(biniptemp,inttostr( porttemp ),inaddrtemp);\r
\r
- If {$ifndef win32}Not{$endif} Bind(fd[a],inAddrtemp,inaddrsize(inaddrtemp)) Then begin\r
- {$ifdef win32}\r
+ fd[a] := Socket(biniptemp.family,SOCK_DGRAM,0);\r
+ bindresult := {$ifdef mswindows}Not{$endif} Bind(fd[a],inAddrtemp,inaddrsize(inaddrtemp));\r
+ dec(retrycount);\r
+ until (retrycount <= 0) or (bindresult);\r
+\r
+ If (not bindresult) Then begin\r
+ {$ifdef mswindows}\r
raise Exception.create('unable to bind '+inttostr(WSAGetLastError));\r
{$else}\r
raise Exception.create('unable to bind '+inttostr(socketError));\r
end;\r
end;\r
\r
-procedure resolveloop(timeout:integer);\r
+procedure cleanupsockets;\r
var\r
- selectresult : integer;\r
- fds : fdset;\r
+ a:integer;\r
+begin\r
+ for a := 0 to numsockused-1 do closesocket(fd[a]);\r
+end;\r
\r
- endtime : longint;\r
- starttime : longint;\r
- wrapmode : boolean;\r
- currenttime : integer;\r
+function sendquery(socknum:integer;const packet:tdnspacket;len:integer):boolean;\r
+var\r
+ ip : tbinip;\r
+ port : ansistring;\r
+ inaddr : TInetSockAddrV;\r
+begin\r
+{ writeln('sendquery ',decodename(state.packet,state.packetlen,12,0,a),' ',state.requesttype);}\r
+ result := false;\r
+ if len = 0 then exit; {no packet}\r
\r
- lag : ttimeval;\r
- selecttimeout : ttimeval;\r
- socknum:integer;\r
- needprocessing:array[0..numsock-1] of boolean;\r
- finished:array[0..numsock-1] of boolean;\r
- a,b:integer;\r
+ ip := getcurrentsystemnameserverbin(id);\r
\r
- Src : TInetSockAddrV;\r
- Srcx : {$ifdef win32}sockaddr_in{$else}TInetSockAddrV{$endif} absolute Src;\r
- SrcLen : Integer;\r
- fromip:tbinip;\r
- fromport:ansistring;\r
+ {$ifdef ipv6}{$ifdef mswindows}\r
+ if toaddr[socknum].family = AF_INET6 then if (useaf = 0) then useaf := useaf_preferv6;\r
+ {$endif}{$endif}\r
+\r
+ port := toport;\r
+ toaddr[socknum] := ip;\r
+ makeinaddrv(toaddr[socknum],port,inaddr);\r
+\r
+ sendto(fd[socknum],packet,len,0,inaddr,inaddrsize(inaddr));\r
+ sendquerytime[socknum] := getts;\r
+ result := true;\r
+end;\r
\r
begin\r
if timeout < mintimeout then timeout := defaulttimeout;\r
\r
- starttime := getts;\r
- endtime := starttime + timeout;\r
- if (endtime and tswrap)=0 then begin\r
- wrapmode := false;\r
- end else begin\r
- wrapmode := true;\r
- end;\r
- endtime := endtime and tsmask;\r
+ starttime := getts;\r
+ endtime := starttime + timeout;\r
+ if (endtime and tswrap)=0 then begin\r
+ wrapmode := false;\r
+ end else begin\r
+ wrapmode := true;\r
+ end;\r
+ endtime := endtime and tsmask;\r
\r
setupsocket;\r
+\r
+\r
for socknum := 0 to numsockused-1 do begin\r
needprocessing[socknum] := true;\r
finished[socknum] := false;\r
if finished[a] then inc(b);\r
end;\r
if (b = numsockused) then begin\r
+ cleanupsockets;\r
exit;\r
end;\r
//onrequestdone(self,0);\r
selecttimeout.tv_sec := 0;\r
selecttimeout.tv_usec := retryafter;\r
end;\r
- //find the highest of the used fd's\r
+ //find the highest of the used fds\r
b := 0;\r
for socknum := numsockused-1 downto 0 do if fd[socknum] > b then b := fd[socknum];\r
selectresult := select(b+1,@fds,nil,nil,@selecttimeout);\r
\r
reportlag(id,-1);\r
if (currenttime >= endtime) and ((not wrapmode) or (currenttime < starttime)) then begin\r
+ cleanupsockets;\r
exit;\r
end else begin\r
//resend\r
a:integer;\r
biniptemp:tbinip;\r
l:tbiniplist;\r
+\r
+ numsockused:integer;\r
+ state:tdnsstatearr;\r
+\r
begin\r
ipstrtobin(name,biniptemp);\r
if biniptemp.family <> 0 then begin\r
exit; //it was an IP address, no need for dns\r
end;\r
\r
- {$ifdef win32}\r
+ {$ifdef mswindows}\r
if usewindns then begin\r
if (useaf = useaf_v4) then a := af_inet else if (useaf = useaf_v6) then a := af_inet6 else a := 0;\r
result := winforwardlookuplist(name,a,dummy);\r
end;\r
{$endif}\r
\r
- resolveloop(timeout);\r
+ resolveloop(timeout,state,numsockused);\r
\r
if (numsockused = 1) then begin\r
biniplist_addlist(result,state[0].resultlist);\r
function reverselookup(ip:tbinip;timeout:integer):ansistring;\r
var\r
dummy : integer;\r
+ numsockused:integer;\r
+ state:tdnsstatearr;\r
begin\r
- {$ifdef win32}\r
+ {$ifdef mswindows}\r
if usewindns then begin\r
result := winreverselookup(ip,dummy);\r
exit;\r
{$ifdef syncdnscore}\r
setstate_reverse(ip,state[0]);\r
numsockused := 1;\r
- resolveloop(timeout);\r
+ resolveloop(timeout,state,numsockused);\r
result := state[0].resultstr;\r
{$endif}\r
end;\r
\r
-{$ifdef win32}\r
+{$ifdef mswindows}\r
var\r
wsadata : twsadata;\r
\r