2 ; -----------------------------------------------------------------------
4 ; Copyright 2004-2008 H. Peter Anvin - All Rights Reserved
6 ; This program is free software; you can redistribute it and/or modify
7 ; it under the terms of the GNU General Public License as published by
8 ; the Free Software Foundation, Inc., 53 Temple Place Ste 330,
9 ; Bostom MA 02111-1307, USA; either version 2 of the License, or
10 ; (at your option) any later version; incorporated herein by reference.
12 ; -----------------------------------------------------------------------
17 ; Very simple DNS resolver (assumes recursion-enabled DNS server;
18 ; this should be the normal thing for client-serving DNS servers.)
21 DNS_PORT equ htons(53) ; Default DNS port
22 DNS_MAX_PACKET equ 512 ; Defined by protocol
23 ; TFTP uses the range 49152-57343
24 DNS_LOCAL_PORT equ htons(60053) ; All local DNS queries come from this port #
25 DNS_MAX_SERVERS equ 4 ; Max no of DNS servers
30 ; Turn a string in DS:SI into a DNS "label set" in ES:DI.
31 ; On return, DI points to the first byte after the label set,
32 ; and SI to the terminating byte.
34 ; On return, DX contains the number of dots encountered.
58 dec dx ; We always counted one high
69 ; Compare two sets of DNS labels, in DS:SI and ES:DI; the one in SI
70 ; is allowed pointers relative to a packet in DNSRecvBuf.
72 ; Assumes DS == ES. ZF = 1 if same; no registers changed.
73 ; (Note: change reference to [di] to [es:di] to remove DS == ES assumption)
83 and al,03Fh ; Get pointer value
84 mov ah,al ; ... in network byte order!
93 movzx cx,al ; End label?
94 and cx,cx ; ZF = 1 if match
97 ; We have a string of bytes that need to match now
109 ; Skip past a DNS label set in DS:SI.
116 cmp al,0C0h ; Pointer?
123 inc si ; Pointer is two bytes
155 DNSSendBuf resb DNS_MAX_PACKET
156 DNSRecvBuf resb DNS_MAX_PACKET
157 LocalDomain resb 256 ; Max possible length
158 DNSServers resd DNS_MAX_SERVERS
161 pxe_udp_write_pkt_dns:
162 .status: dw 0 ; Status
163 .sip: dd 0 ; Server IP
164 .gip: dd 0 ; Gateway IP
165 .lport: dw DNS_LOCAL_PORT ; Local port
166 .rport: dw DNS_PORT ; Remote port
167 .buffersize: dw 0 ; Size of packet
168 .buffer: dw DNSSendBuf, 0 ; off, seg of buffer
170 pxe_udp_read_pkt_dns:
171 .status: dw 0 ; Status
172 .sip: dd 0 ; Source IP
173 .dip: dd 0 ; Destination (our) IP
174 .rport: dw DNS_PORT ; Remote port
175 .lport: dw DNS_LOCAL_PORT ; Local port
176 .buffersize: dw DNS_MAX_PACKET ; Max packet size
177 .buffer: dw DNSRecvBuf, 0 ; off, seg of buffer
179 LastDNSServer dw DNSServers
181 ; Actual resolver function
182 ; Points to a null-terminated or :-terminated string in DS:SI
183 ; and returns the name in EAX if it exists and can be found.
184 ; If EAX = 0 on exit, the lookup failed.
186 ; No segment assumptions permitted.
200 ; First, build query packet
201 mov di,DNSSendBuf+dnshdr.flags
202 inc word [es:di-2] ; New query ID
203 mov ax,htons(0100h) ; Recursion requested
205 mov ax,htons(1) ; One question
207 xor ax,ax ; No answers, NS or ARs
212 call dns_mangle ; Convert name to DNS labels
217 push si ; Save pointer to after DNS string
221 mov [pxe_udp_read_pkt_dns.dip],eax
224 jnz .fqdn ; If we have dots, assume it's FQDN
225 dec di ; Remove final null
227 call strcpy ; Uncompressed DNS label set so it ends in null
231 stosw ; QTYPE = 1 = A
232 stosw ; QCLASS = 1 = IN
235 mov [pxe_udp_write_pkt_dns.buffersize],di
237 ; Now, send it to the nameserver(s)
238 ; Outer loop: exponential backoff
239 ; Inner loop: scan the various DNS servers
246 cmp si,[LastDNSServer]
251 cmp bx,TimeoutTableEnd
254 xor eax,eax ; Nothing...
266 lodsd ; EAX <- next server
272 mov word [pxe_udp_write_pkt_dns.status],0
274 mov [pxe_udp_write_pkt_dns.sip],eax
275 mov [pxe_udp_read_pkt_dns.sip],eax
281 mov [pxe_udp_write_pkt_dns.gip],eax
283 mov di,pxe_udp_write_pkt_dns
284 mov bx,PXENV_UDP_WRITE
286 jc .timeout ; Treat failed transmit as timeout
287 cmp word [pxe_udp_write_pkt_dns.status],0
297 mov word [pxe_udp_read_pkt_dns.status],0
298 mov word [pxe_udp_read_pkt_dns.buffersize],DNS_MAX_PACKET
299 mov di,pxe_udp_read_pkt_dns
300 mov bx,PXENV_UDP_READ
304 cmp [pxe_udp_read_pkt_dns.status],ax
307 ; Got a packet, deal with it...
310 cmp ax,[DNSSendBuf] ; ID
311 jne .waitrecv ; Not ours
314 xor al,80h ; Query#/Answer bit
315 test ax,htons(0F80Fh)
320 mov cx,ax ; Questions echoed
325 lodsw ; Authority records
329 call dns_skiplabel ; Skip name
330 add si,4 ; Skip question trailer
334 pop cx ; Number of replies
338 mov di,DNSSendBuf+dnshdr_size
342 mov ax,[si+8] ; RDLENGTH
346 cmp dword [si],htons(1)*0x10001 ; TYPE = A, CLASS = IN?
348 cmp ax,4 ; RDLENGTH = 4?
351 ; We hit paydirt here...
355 add sp,8 ; Drop timeout information
364 ; We got back no data from this server.
365 ; Unfortunately, for a recursive, non-authoritative
366 ; query there is no such thing as an NXDOMAIN reply,
367 ; which technically means we can't draw any
368 ; conclusions. However, in practice that means the
369 ; domain doesn't exist. If this turns out to be a
370 ; problem, we may want to add code to go through all
371 ; the servers before giving up.
373 ; If the DNS server wasn't capable of recursion, and
374 ; isn't capable of giving us an authoritative reply
375 ; (i.e. neither AA or RA set), then at least try a
376 ; different setver...
378 test word [DNSRecvBuf+dnshdr.flags],htons(0480h)