2 * COPYRIGHT: See COPYING in the top level directory
3 * PROJECT: ReactOS system libraries
4 * FILE: lib/dnsapi/dnsapi/query.c
5 * PURPOSE: DNSAPI functions built on the ADNS library.
6 * PROGRAMER: Art Yerkes
16 /* DnsQuery ****************************
17 * Begin a DNS query, and allow the result to be placed in the application
18 * supplied result pointer. The result can be manipulated with the record
21 * Name -- The DNS object to be queried.
22 * Type -- The type of records to be returned. These are
24 * Options -- Query options. DNS_QUERY_STANDARD is the base
25 * state, and every other option takes precedence.
26 * multiple options can be combined. Listed in
28 * Servers -- List of alternate servers (optional)
29 * QueryResultSet -- Pointer to the result pointer that will be filled
30 * when the response is available.
31 * Reserved -- Response as it appears on the wire. Optional.
35 *xstrsave(const char *str
)
39 p
= RtlAllocateHeap(RtlGetProcessHeap(), 0, strlen(str
) + 1);
48 DnsQuery_A(LPCSTR Name
,
52 PDNS_RECORD
*QueryResultSet
,
67 adns_error
= adns_init(&astate
, adns_if_noenv
| adns_if_noerrprint
| adns_if_noserverwarn
| (Servers
? adns_if_noserver
: 0), 0);
69 if(adns_error
!= adns_s_ok
)
70 return DnsIntTranslateAdnsToDNS_STATUS(adns_error
);
74 for(i
= 0; i
< Servers
->AddrCount
; i
++)
76 adns_addserver(astate
, *((struct in_addr
*)&Servers
->AddrArray
[i
]));
81 * adns doesn't resolve chained CNAME records (a CNAME which points to
82 * another CNAME pointing to another... pointing to an A record), according
83 * to a mailing list thread the authors believe that chained CNAME records
84 * are invalid and the DNS entries should be fixed. That's a nice academic
85 * standpoint, but there certainly are chained CNAME records out there,
86 * even some fairly major ones (at the time of this writing
87 * download.mozilla.org is a chained CNAME). Everyone else seems to resolve
88 * these fine, so we should too. So we loop here to try to resolve CNAME
89 * chains ourselves. Of course, there must be a limit to protect against
93 #define CNAME_LOOP_MAX 16
95 CurrentName
= (LPSTR
) Name
;
97 for (CNameLoop
= 0; CNameLoop
< CNAME_LOOP_MAX
; CNameLoop
++)
99 adns_error
= adns_synchronous(astate
, CurrentName
, adns_r_addr
, quflags
, &answer
);
101 if(adns_error
!= adns_s_ok
)
105 if (CurrentName
!= Name
)
106 RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName
);
108 return DnsIntTranslateAdnsToDNS_STATUS(adns_error
);
111 if(answer
&& answer
->rrs
.addr
)
113 if (CurrentName
!= Name
)
114 RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName
);
116 *QueryResultSet
= (PDNS_RECORD
)RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_RECORD
));
118 if (NULL
== *QueryResultSet
)
120 adns_finish( astate
);
121 return ERROR_OUTOFMEMORY
;
124 (*QueryResultSet
)->pNext
= NULL
;
125 (*QueryResultSet
)->wType
= Type
;
126 (*QueryResultSet
)->wDataLength
= sizeof(DNS_A_DATA
);
127 (*QueryResultSet
)->Data
.A
.IpAddress
= answer
->rrs
.addr
->addr
.inet
.sin_addr
.s_addr
;
131 (*QueryResultSet
)->pName
= xstrsave( Name
);
133 return NULL
!= (*QueryResultSet
)->pName
? ERROR_SUCCESS
: ERROR_OUTOFMEMORY
;
136 if (NULL
== answer
|| adns_s_prohibitedcname
!= answer
->status
|| NULL
== answer
->cname
)
140 if (CurrentName
!= Name
)
141 RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName
);
143 return ERROR_FILE_NOT_FOUND
;
146 if (CurrentName
!= Name
)
147 RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName
);
149 CurrentName
= xstrsave(answer
->cname
);
154 return ERROR_OUTOFMEMORY
;
159 RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName
);
160 return ERROR_FILE_NOT_FOUND
;
163 return ERROR_OUTOFMEMORY
; /* XXX arty: find a better error code. */
168 DnsWToC(const WCHAR
*WideString
)
170 int chars
= wcstombs(NULL
, WideString
, 0);
171 PCHAR out
= RtlAllocateHeap(RtlGetProcessHeap(), 0, chars
+ 1);
173 wcstombs(out
, WideString
, chars
+ 1);
179 DnsCToW(const CHAR
*NarrowString
)
181 int chars
= mbstowcs(NULL
, NarrowString
, 0);
182 PWCHAR out
= RtlAllocateHeap(RtlGetProcessHeap(), 0, (chars
+ 1) * sizeof(WCHAR
));
184 mbstowcs(out
, NarrowString
, chars
+ 1);
190 DnsQuery_W(LPCWSTR Name
,
194 PDNS_RECORD
*QueryResultSet
,
200 PDNS_RECORD QueryResultWide
;
201 PDNS_RECORD ConvertedRecord
= 0, LastRecord
= 0;
203 Buffer
= DnsWToC(Name
);
205 Status
= DnsQuery_A(Buffer
, Type
, Options
, Servers
, &QueryResultWide
, Reserved
);
207 while(Status
== ERROR_SUCCESS
&& QueryResultWide
)
209 switch(QueryResultWide
->wType
)
213 ConvertedRecord
= RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_RECORD
));
214 ConvertedRecord
->pName
= (PCHAR
)DnsCToW(QueryResultWide
->pName
);
215 ConvertedRecord
->wType
= QueryResultWide
->wType
;
216 ConvertedRecord
->wDataLength
= QueryResultWide
->wDataLength
;
217 memcpy(ConvertedRecord
, QueryResultWide
, QueryResultWide
->wDataLength
);
228 ConvertedRecord
= RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_RECORD
));
229 ConvertedRecord
->pName
= (PCHAR
)DnsCToW(QueryResultWide
->pName
);
230 ConvertedRecord
->wType
= QueryResultWide
->wType
;
231 ConvertedRecord
->wDataLength
= sizeof(DNS_PTR_DATA
);
232 ConvertedRecord
->Data
.PTR
.pNameHost
= (PCHAR
)DnsCToW(QueryResultWide
->Data
.PTR
.pNameHost
);
236 ConvertedRecord
= RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_RECORD
));
237 ConvertedRecord
->pName
= (PCHAR
)DnsCToW(QueryResultWide
->pName
);
238 ConvertedRecord
->wType
= QueryResultWide
->wType
;
239 ConvertedRecord
->wDataLength
= sizeof(DNS_MINFO_DATA
);
240 ConvertedRecord
->Data
.MINFO
.pNameMailbox
= (PCHAR
)DnsCToW(QueryResultWide
->Data
.MINFO
.pNameMailbox
);
241 ConvertedRecord
->Data
.MINFO
.pNameErrorsMailbox
= (PCHAR
)DnsCToW(QueryResultWide
->Data
.MINFO
.pNameErrorsMailbox
);
245 ConvertedRecord
= RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_RECORD
));
246 ConvertedRecord
->pName
= (PCHAR
)DnsCToW(QueryResultWide
->pName
);
247 ConvertedRecord
->wType
= QueryResultWide
->wType
;
248 ConvertedRecord
->wDataLength
= sizeof(DNS_MX_DATA
);
249 ConvertedRecord
->Data
.MX
.pNameExchange
= (PCHAR
)DnsCToW( QueryResultWide
->Data
.MX
.pNameExchange
);
250 ConvertedRecord
->Data
.MX
.wPreference
= QueryResultWide
->Data
.MX
.wPreference
;
254 ConvertedRecord
= RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_TXT_DATA
) + QueryResultWide
->Data
.TXT
.dwStringCount
);
255 ConvertedRecord
->pName
= (PCHAR
)DnsCToW( QueryResultWide
->pName
);
256 ConvertedRecord
->wType
= QueryResultWide
->wType
;
257 ConvertedRecord
->wDataLength
= sizeof(DNS_TXT_DATA
) + (sizeof(PWCHAR
) * QueryResultWide
->Data
.TXT
.dwStringCount
);
258 ConvertedRecord
->Data
.TXT
.dwStringCount
= QueryResultWide
->Data
.TXT
.dwStringCount
;
260 for(i
= 0; i
< ConvertedRecord
->Data
.TXT
.dwStringCount
; i
++)
261 ConvertedRecord
->Data
.TXT
.pStringArray
[i
] = (PCHAR
)DnsCToW(QueryResultWide
->Data
.TXT
.pStringArray
[i
]);
266 ConvertedRecord
= RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_NULL_DATA
) + QueryResultWide
->Data
.Null
.dwByteCount
);
267 ConvertedRecord
->pName
= (PCHAR
)DnsCToW(QueryResultWide
->pName
);
268 ConvertedRecord
->wType
= QueryResultWide
->wType
;
269 ConvertedRecord
->wDataLength
= sizeof(DNS_NULL_DATA
) + QueryResultWide
->Data
.Null
.dwByteCount
;
270 ConvertedRecord
->Data
.Null
.dwByteCount
= QueryResultWide
->Data
.Null
.dwByteCount
;
271 memcpy(&ConvertedRecord
->Data
.Null
.Data
, &QueryResultWide
->Data
.Null
.Data
, QueryResultWide
->Data
.Null
.dwByteCount
);
277 LastRecord
->pNext
= ConvertedRecord
;
278 LastRecord
= LastRecord
->pNext
;
282 LastRecord
= *QueryResultSet
= ConvertedRecord
;
286 LastRecord
->pNext
= 0;
289 RtlFreeHeap(RtlGetProcessHeap(), 0, Buffer
);
295 DnsQuery_UTF8(LPCSTR Name
,
299 PDNS_RECORD
*QueryResultSet
,
303 return ERROR_OUTOFMEMORY
;
307 DnsIntFreeRecordList(PDNS_RECORD ToDelete
)
310 PDNS_RECORD next
= 0;
315 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete
->pName
);
317 switch(ToDelete
->wType
)
327 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete
->Data
.PTR
.pNameHost
);
332 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete
->Data
.MX
.pNameExchange
);
336 for(i
= 0; i
< ToDelete
->Data
.TXT
.dwStringCount
; i
++)
337 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete
->Data
.TXT
.pStringArray
[i
]);
339 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete
->Data
.TXT
.pStringArray
);
343 next
= ToDelete
->pNext
;
344 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete
);