xref: /reactos/dll/win32/dnsapi/query.c (revision 40462c92)
1 /*
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
7  * UPDATE HISTORY:
8  *              12/15/03 -- Created
9  */
10 
11 #include "precomp.h"
12 #include <winreg.h>
13 #include <iphlpapi.h>
14 #include <strsafe.h>
15 
16 #define NDEBUG
17 #include <debug.h>
18 
19 static
20 BOOL
21 ParseIpv4Address(
22     _In_ PCWSTR AddressString,
23     _Out_ PIN_ADDR pAddress)
24 {
25     PCWSTR pTerminator = NULL;
26     NTSTATUS Status;
27 
28     Status = RtlIpv4StringToAddressW(AddressString,
29                                      TRUE,
30                                      &pTerminator,
31                                      pAddress);
32     if (NT_SUCCESS(Status) && pTerminator != NULL && *pTerminator == L'\0')
33         return TRUE;
34 
35     return FALSE;
36 }
37 
38 
39 static
40 BOOL
41 ParseIpv6Address(
42     _In_ PCWSTR AddressString,
43     _Out_ PIN6_ADDR pAddress)
44 {
45     PCWSTR pTerminator = NULL;
46     NTSTATUS Status;
47 
48     Status = RtlIpv6StringToAddressW(AddressString,
49                                      &pTerminator,
50                                      pAddress);
51     if (NT_SUCCESS(Status) && pTerminator != NULL && *pTerminator == L'\0')
52         return TRUE;
53 
54     return FALSE;
55 }
56 
57 
58 static
59 PDNS_RECORDW
60 CreateRecordForIpAddress(
61     _In_ PCWSTR Name,
62     _In_ WORD Type)
63 {
64     IN_ADDR Ip4Address;
65     IN6_ADDR Ip6Address;
66     PDNS_RECORDW pRecord = NULL;
67 
68     if (Type == DNS_TYPE_A)
69     {
70         if (ParseIpv4Address(Name, &Ip4Address))
71         {
72             pRecord = RtlAllocateHeap(RtlGetProcessHeap(),
73                                       HEAP_ZERO_MEMORY,
74                                       sizeof(DNS_RECORDW));
75             if (pRecord == NULL)
76                 return NULL;
77 
78             pRecord->pName = RtlAllocateHeap(RtlGetProcessHeap(),
79                                              0,
80                                              (wcslen(Name) + 1) * sizeof(WCHAR));
81             if (pRecord == NULL)
82             {
83                 RtlFreeHeap(RtlGetProcessHeap(), 0, pRecord);
84                 return NULL;
85             }
86 
87             wcscpy(pRecord->pName, Name);
88             pRecord->wType = DNS_TYPE_A;
89             pRecord->wDataLength = sizeof(DNS_A_DATA);
90             pRecord->Flags.S.Section = DnsSectionQuestion;
91             pRecord->Flags.S.CharSet = DnsCharSetUnicode;
92             pRecord->dwTtl = 7 * 24 * 60 * 60;
93 
94             pRecord->Data.A.IpAddress = Ip4Address.S_un.S_addr;
95 
96             return pRecord;
97         }
98     }
99     else if (Type == DNS_TYPE_AAAA)
100     {
101         if (ParseIpv6Address(Name, &Ip6Address))
102         {
103             pRecord = RtlAllocateHeap(RtlGetProcessHeap(),
104                                       HEAP_ZERO_MEMORY,
105                                       sizeof(DNS_RECORDW));
106             if (pRecord == NULL)
107                 return NULL;
108 
109             pRecord->pName = RtlAllocateHeap(RtlGetProcessHeap(),
110                                              0,
111                                              (wcslen(Name) + 1) * sizeof(WCHAR));
112             if (pRecord == NULL)
113             {
114                 RtlFreeHeap(RtlGetProcessHeap(), 0, pRecord);
115                 return NULL;
116             }
117 
118             wcscpy(pRecord->pName, Name);
119             pRecord->wType = DNS_TYPE_AAAA;
120             pRecord->wDataLength = sizeof(DNS_AAAA_DATA);
121             pRecord->Flags.S.Section = DnsSectionQuestion;
122             pRecord->Flags.S.CharSet = DnsCharSetUnicode;
123             pRecord->dwTtl = 7 * 24 * 60 * 60;
124 
125             CopyMemory(&pRecord->Data.AAAA.Ip6Address,
126                        &Ip6Address.u.Byte,
127                        sizeof(IN6_ADDR));
128 
129             return pRecord;
130         }
131     }
132 
133     return NULL;
134 }
135 
136 
137 /* DnsQuery ****************************
138  * Begin a DNS query, and allow the result to be placed in the application
139  * supplied result pointer.  The result can be manipulated with the record
140  * functions.
141  *
142  * Name                 -- The DNS object to be queried.
143  * Type                 -- The type of records to be returned.  These are
144  *                          listed in windns.h
145  * Options              -- Query options.  DNS_QUERY_STANDARD is the base
146  *                          state, and every other option takes precedence.
147  *                          multiple options can be combined.  Listed in
148  *                          windns.h
149  * Servers              -- List of alternate servers (optional)
150  * QueryResultSet       -- Pointer to the result pointer that will be filled
151  *                          when the response is available.
152  * Reserved             -- Response as it appears on the wire.  Optional.
153  */
154 
155 static PCHAR
156 DnsWToC(const WCHAR *WideString)
157 {
158     PCHAR AnsiString;
159     int AnsiLen = WideCharToMultiByte(CP_ACP,
160                                       0,
161                                       WideString,
162                                       -1,
163                                       NULL,
164                                       0,
165                                       NULL,
166                                       0);
167     if (AnsiLen == 0)
168         return NULL;
169     AnsiString = RtlAllocateHeap(RtlGetProcessHeap(), 0, AnsiLen);
170     if (AnsiString == NULL)
171     {
172         return NULL;
173     }
174     WideCharToMultiByte(CP_ACP,
175                         0,
176                         WideString,
177                         -1,
178                         AnsiString,
179                         AnsiLen,
180                         NULL,
181                         0);
182 
183     return AnsiString;
184 }
185 
186 static PWCHAR
187 DnsCToW(const CHAR *NarrowString)
188 {
189     PWCHAR WideString;
190     int WideLen = MultiByteToWideChar(CP_ACP,
191                                       0,
192                                       NarrowString,
193                                       -1,
194                                       NULL,
195                                       0);
196     if (WideLen == 0)
197         return NULL;
198     WideString = RtlAllocateHeap(RtlGetProcessHeap(), 0, WideLen * sizeof(WCHAR));
199     if (WideString == NULL)
200     {
201         return NULL;
202     }
203     MultiByteToWideChar(CP_ACP,
204                         0,
205                         NarrowString,
206                         -1,
207                         WideString,
208                         WideLen);
209 
210     return WideString;
211 }
212 
213 static PCHAR
214 DnsWToUTF8(const WCHAR *WideString)
215 {
216     PCHAR AnsiString;
217     int AnsiLen = WideCharToMultiByte(CP_UTF8,
218                                       0,
219                                       WideString,
220                                       -1,
221                                       NULL,
222                                       0,
223                                       NULL,
224                                       0);
225     if (AnsiLen == 0)
226         return NULL;
227     AnsiString = RtlAllocateHeap(RtlGetProcessHeap(), 0, AnsiLen);
228     if (AnsiString == NULL)
229     {
230         return NULL;
231     }
232     WideCharToMultiByte(CP_UTF8,
233                         0,
234                         WideString,
235                         -1,
236                         AnsiString,
237                         AnsiLen,
238                         NULL,
239                         0);
240 
241     return AnsiString;
242 }
243 
244 static PWCHAR
245 DnsUTF8ToW(const CHAR *NarrowString)
246 {
247     PWCHAR WideString;
248     int WideLen = MultiByteToWideChar(CP_UTF8,
249                                       0,
250                                       NarrowString,
251                                       -1,
252                                       NULL,
253                                       0);
254     if (WideLen == 0)
255         return NULL;
256     WideString = RtlAllocateHeap(RtlGetProcessHeap(), 0, WideLen * sizeof(WCHAR));
257     if (WideString == NULL)
258     {
259         return NULL;
260     }
261     MultiByteToWideChar(CP_UTF8,
262                         0,
263                         NarrowString,
264                         -1,
265                         WideString,
266                         WideLen);
267 
268     return WideString;
269 }
270 
271 DNS_STATUS WINAPI
272 DnsQuery_CodePage(UINT CodePage,
273            LPCSTR Name,
274            WORD Type,
275            DWORD Options,
276            PVOID Extra,
277            PDNS_RECORD *QueryResultSet,
278            PVOID *Reserved)
279 {
280     UINT i;
281     PWCHAR Buffer;
282     DNS_STATUS Status;
283     PDNS_RECORD QueryResultWide;
284     PDNS_RECORD ConvertedRecord = 0, LastRecord = 0;
285 
286     if (Name == NULL)
287         return ERROR_INVALID_PARAMETER;
288     if (QueryResultSet == NULL)
289         return ERROR_INVALID_PARAMETER;
290 
291     switch (CodePage)
292     {
293     case CP_ACP:
294         Buffer = DnsCToW(Name);
295         break;
296 
297     case CP_UTF8:
298         Buffer = DnsUTF8ToW(Name);
299         break;
300 
301     default:
302         return ERROR_INVALID_PARAMETER;
303     }
304 
305     Status = DnsQuery_W(Buffer, Type, Options, Extra, &QueryResultWide, Reserved);
306 
307     while (Status == ERROR_SUCCESS && QueryResultWide)
308     {
309         switch (QueryResultWide->wType)
310         {
311         case DNS_TYPE_A:
312         case DNS_TYPE_WKS:
313         case DNS_TYPE_CNAME:
314         case DNS_TYPE_PTR:
315         case DNS_TYPE_NS:
316         case DNS_TYPE_MB:
317         case DNS_TYPE_MD:
318         case DNS_TYPE_MF:
319         case DNS_TYPE_MG:
320         case DNS_TYPE_MR:
321             ConvertedRecord = RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_RECORD));
322             break;
323 
324         case DNS_TYPE_MINFO:
325             ConvertedRecord = RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_TXT_DATA) + QueryResultWide->Data.TXT.dwStringCount);
326             break;
327 
328         case DNS_TYPE_NULL:
329             ConvertedRecord = RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_NULL_DATA) + QueryResultWide->Data.Null.dwByteCount);
330             break;
331         }
332         if (ConvertedRecord == NULL)
333         {
334             /* The name */
335             RtlFreeHeap(RtlGetProcessHeap(), 0, Buffer);
336             /* The result*/
337             DnsIntFreeRecordList(QueryResultWide);
338             QueryResultSet = NULL;
339             return ERROR_OUTOFMEMORY;
340         }
341 
342         if (CodePage == CP_ACP)
343         {
344             ConvertedRecord->pName = DnsWToC((PWCHAR)QueryResultWide->pName);
345             ConvertedRecord->Flags.S.CharSet = DnsCharSetAnsi;
346         }
347         else
348         {
349             ConvertedRecord->pName = DnsWToUTF8((PWCHAR)QueryResultWide->pName);
350             ConvertedRecord->Flags.S.CharSet = DnsCharSetUtf8;
351         }
352 
353         ConvertedRecord->wType = QueryResultWide->wType;
354 
355         switch (QueryResultWide->wType)
356         {
357         case DNS_TYPE_A:
358         case DNS_TYPE_WKS:
359             ConvertedRecord->wDataLength = QueryResultWide->wDataLength;
360             memcpy(&ConvertedRecord->Data, &QueryResultWide->Data, QueryResultWide->wDataLength);
361             break;
362 
363         case DNS_TYPE_CNAME:
364         case DNS_TYPE_PTR:
365         case DNS_TYPE_NS:
366         case DNS_TYPE_MB:
367         case DNS_TYPE_MD:
368         case DNS_TYPE_MF:
369         case DNS_TYPE_MG:
370         case DNS_TYPE_MR:
371             ConvertedRecord->wDataLength = sizeof(DNS_PTR_DATA);
372             if (CodePage == CP_ACP)
373                 ConvertedRecord->Data.PTR.pNameHost = DnsWToC((PWCHAR)QueryResultWide->Data.PTR.pNameHost);
374             else
375                 ConvertedRecord->Data.PTR.pNameHost = DnsWToUTF8((PWCHAR)QueryResultWide->Data.PTR.pNameHost);
376             break;
377 
378         case DNS_TYPE_MINFO:
379             ConvertedRecord->wDataLength = sizeof(DNS_MINFO_DATA);
380             if (CodePage == CP_ACP)
381             {
382                 ConvertedRecord->Data.MINFO.pNameMailbox = DnsWToC((PWCHAR)QueryResultWide->Data.MINFO.pNameMailbox);
383                 ConvertedRecord->Data.MINFO.pNameErrorsMailbox = DnsWToC((PWCHAR)QueryResultWide->Data.MINFO.pNameErrorsMailbox);
384             }
385             else
386             {
387                 ConvertedRecord->Data.MINFO.pNameMailbox = DnsWToUTF8((PWCHAR)QueryResultWide->Data.MINFO.pNameMailbox);
388                 ConvertedRecord->Data.MINFO.pNameErrorsMailbox = DnsWToUTF8((PWCHAR)QueryResultWide->Data.MINFO.pNameErrorsMailbox);
389             }
390             break;
391 
392         case DNS_TYPE_MX:
393             ConvertedRecord->wDataLength = sizeof(DNS_MX_DATA);
394             if (CodePage == CP_ACP)
395                 ConvertedRecord->Data.MX.pNameExchange = DnsWToC((PWCHAR)QueryResultWide->Data.MX.pNameExchange);
396             else
397                 ConvertedRecord->Data.MX.pNameExchange = DnsWToUTF8((PWCHAR)QueryResultWide->Data.MX.pNameExchange);
398             ConvertedRecord->Data.MX.wPreference = QueryResultWide->Data.MX.wPreference;
399             break;
400 
401         case DNS_TYPE_HINFO:
402             ConvertedRecord->wDataLength = sizeof(DNS_TXT_DATA) + (sizeof(PCHAR) * QueryResultWide->Data.TXT.dwStringCount);
403             ConvertedRecord->Data.TXT.dwStringCount = QueryResultWide->Data.TXT.dwStringCount;
404 
405             if (CodePage == CP_ACP)
406                 for (i = 0; i < ConvertedRecord->Data.TXT.dwStringCount; i++)
407                     ConvertedRecord->Data.TXT.pStringArray[i] = DnsWToC((PWCHAR)QueryResultWide->Data.TXT.pStringArray[i]);
408             else
409                 for (i = 0; i < ConvertedRecord->Data.TXT.dwStringCount; i++)
410                     ConvertedRecord->Data.TXT.pStringArray[i] = DnsWToUTF8((PWCHAR)QueryResultWide->Data.TXT.pStringArray[i]);
411 
412             break;
413 
414         case DNS_TYPE_NULL:
415             ConvertedRecord->wDataLength = sizeof(DNS_NULL_DATA) + QueryResultWide->Data.Null.dwByteCount;
416             ConvertedRecord->Data.Null.dwByteCount = QueryResultWide->Data.Null.dwByteCount;
417             memcpy(&ConvertedRecord->Data.Null.Data, &QueryResultWide->Data.Null.Data, QueryResultWide->Data.Null.dwByteCount);
418             break;
419         }
420 
421         if (LastRecord)
422         {
423             LastRecord->pNext = ConvertedRecord;
424             LastRecord = LastRecord->pNext;
425         }
426         else
427         {
428             LastRecord = *QueryResultSet = ConvertedRecord;
429         }
430 
431         QueryResultWide = QueryResultWide->pNext;
432     }
433 
434     if (LastRecord)
435         LastRecord->pNext = 0;
436 
437     /* The name */
438     RtlFreeHeap(RtlGetProcessHeap(), 0, Buffer);
439     /* The result*/
440     if (QueryResultWide) DnsIntFreeRecordList(QueryResultWide);
441 
442     return Status;
443 }
444 
445 DNS_STATUS WINAPI
446 DnsQuery_A(LPCSTR Name,
447            WORD Type,
448            DWORD Options,
449            PVOID Extra,
450            PDNS_RECORD *QueryResultSet,
451            PVOID *Reserved)
452 {
453     return DnsQuery_CodePage(CP_ACP, Name, Type, Options, Extra, QueryResultSet, Reserved);
454 }
455 
456 DNS_STATUS WINAPI
457 DnsQuery_UTF8(LPCSTR Name,
458               WORD Type,
459               DWORD Options,
460               PVOID Extra,
461               PDNS_RECORD *QueryResultSet,
462               PVOID *Reserved)
463 {
464     return DnsQuery_CodePage(CP_UTF8, Name, Type, Options, Extra, QueryResultSet, Reserved);
465 }
466 
467 DNS_STATUS
468 WINAPI
469 DnsQuery_W(LPCWSTR Name,
470            WORD Type,
471            DWORD Options,
472            PVOID Extra,
473            PDNS_RECORD *QueryResultSet,
474            PVOID *Reserved)
475 {
476     DWORD dwRecords = 0;
477     PDNS_RECORDW pRecord = NULL;
478     size_t NameLen, i;
479     DNS_STATUS Status = ERROR_SUCCESS;
480 
481     DPRINT("DnsQuery_W()\n");
482 
483     if ((Name == NULL) ||
484         (QueryResultSet == NULL))
485         return ERROR_INVALID_PARAMETER;
486 
487     *QueryResultSet = NULL;
488 
489     /* Create an A or AAAA record for an IP4 or IP6 address */
490     pRecord = CreateRecordForIpAddress(Name,
491                                        Type);
492     if (pRecord != NULL)
493     {
494         *QueryResultSet = (PDNS_RECORD)pRecord;
495         return ERROR_SUCCESS;
496     }
497 
498     /*
499      * Check allowed characters
500      * According to RFC a-z,A-Z,0-9,-,_, but can't start or end with - or _
501      */
502     NameLen = wcslen(Name);
503     if (Name[0] == L'-' || Name[0] == L'_' || Name[NameLen - 1] == L'-' ||
504         Name[NameLen - 1] == L'_' || wcsstr(Name, L"..") != NULL)
505     {
506         return ERROR_INVALID_NAME;
507     }
508 
509     i = 0;
510     while (i < NameLen)
511     {
512         if (!((Name[i] >= L'a' && Name[i] <= L'z') ||
513               (Name[i] >= L'A' && Name[i] <= L'Z') ||
514               (Name[i] >= L'0' && Name[i] <= L'9') ||
515               Name[i] == L'-' || Name[i] == L'_' || Name[i] == L'.'))
516         {
517             return DNS_ERROR_INVALID_NAME_CHAR;
518         }
519 
520         i++;
521     }
522 
523     RpcTryExcept
524     {
525         Status = R_ResolverQuery(NULL,
526                                  Name,
527                                  Type,
528                                  Options,
529                                  &dwRecords,
530                                  (DNS_RECORDW **)QueryResultSet);
531         DPRINT("R_ResolverQuery() returned %lu\n", Status);
532     }
533     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
534     {
535         Status = RpcExceptionCode();
536         DPRINT("Exception returned %lu\n", Status);
537     }
538     RpcEndExcept;
539 
540     return Status;
541 }
542 
543 WCHAR
544 *xstrsave(const WCHAR *str)
545 {
546     WCHAR *p;
547     size_t len = 0;
548 
549     /* FIXME: how much instead of MAX_PATH? */
550     StringCbLengthW(str, MAX_PATH, &len);
551     len+=sizeof(WCHAR);
552 
553     p = RtlAllocateHeap(RtlGetProcessHeap(), 0, len);
554 
555     if (p)
556         StringCbCopyW(p, len, str);
557 
558     return p;
559 }
560 
561 CHAR
562 *xstrsaveA(const CHAR *str)
563 {
564     CHAR *p;
565     size_t len = 0;
566 
567     /* FIXME: how much instead of MAX_PATH? */
568     StringCbLengthA(str, MAX_PATH, &len);
569     len++;
570 
571     p = RtlAllocateHeap(RtlGetProcessHeap(), 0, len);
572 
573     if (p)
574         StringCbCopyA(p, len, str);
575 
576     return p;
577 }
578 
579 
580 /* This function is far from perfect but it works enough */
581 IP4_ADDRESS
582 CheckForCurrentHostname(CONST CHAR * Name, PFIXED_INFO network_info)
583 {
584     PCHAR TempName;
585     DWORD AdapterAddressesSize, Status;
586     IP4_ADDRESS ret = 0, Address;
587     PIP_ADAPTER_ADDRESSES Addresses = NULL, pip;
588     BOOL Found = FALSE;
589 
590     if (network_info->DomainName[0])
591     {
592         size_t StringLength;
593         size_t TempSize = 2;
594         StringCchLengthA(network_info->HostName, sizeof(network_info->HostName), &StringLength);
595         TempSize += StringLength;
596         StringCchLengthA(network_info->DomainName, sizeof(network_info->DomainName), &StringLength);
597         TempSize += StringLength;
598         TempName = RtlAllocateHeap(RtlGetProcessHeap(), 0, TempSize);
599         StringCchCopyA(TempName, TempSize, network_info->HostName);
600         StringCchCatA(TempName, TempSize, ".");
601         StringCchCatA(TempName, TempSize, network_info->DomainName);
602     }
603     else
604     {
605         TempName = RtlAllocateHeap(RtlGetProcessHeap(), 0, 1);
606         TempName[0] = 0;
607     }
608     Found = !stricmp(Name, network_info->HostName) || !stricmp(Name, TempName);
609     RtlFreeHeap(RtlGetProcessHeap(), 0, TempName);
610     if (!Found)
611     {
612         return 0;
613     }
614     /* get adapter info */
615     AdapterAddressesSize = 0;
616     GetAdaptersAddresses(AF_INET,
617                          GAA_FLAG_SKIP_FRIENDLY_NAME | GAA_FLAG_SKIP_DNS_SERVER |
618                          GAA_FLAG_SKIP_ANYCAST | GAA_FLAG_SKIP_MULTICAST,
619                          NULL,
620                          Addresses,
621                          &AdapterAddressesSize);
622     if (!AdapterAddressesSize)
623     {
624         return 0;
625     }
626     Addresses = RtlAllocateHeap(RtlGetProcessHeap(), 0, AdapterAddressesSize);
627     Status = GetAdaptersAddresses(AF_INET,
628                                   GAA_FLAG_SKIP_FRIENDLY_NAME | GAA_FLAG_SKIP_DNS_SERVER |
629                                   GAA_FLAG_SKIP_ANYCAST | GAA_FLAG_SKIP_MULTICAST,
630                                   NULL,
631                                   Addresses,
632                                   &AdapterAddressesSize);
633     if (Status)
634     {
635         RtlFreeHeap(RtlGetProcessHeap(), 0, Addresses);
636         return 0;
637     }
638     for (pip = Addresses; pip != NULL; pip = pip->Next) {
639         Address = ((LPSOCKADDR_IN)pip->FirstUnicastAddress->Address.lpSockaddr)->sin_addr.S_un.S_addr;
640         if (Address != ntohl(INADDR_LOOPBACK))
641             break;
642     }
643     if (Address && Address != ntohl(INADDR_LOOPBACK))
644     {
645         ret = Address;
646     }
647     RtlFreeHeap(RtlGetProcessHeap(), 0, Addresses);
648     return ret;
649 }
650 
651 
652 DNS_STATUS
653 WINAPI
654 Query_Main(LPCWSTR Name,
655            WORD Type,
656            DWORD Options,
657            PDNS_RECORD *QueryResultSet)
658 {
659     adns_state astate;
660     int quflags = (Options & DNS_QUERY_NO_RECURSION) == 0 ? adns_qf_search : 0;
661     int adns_error;
662     adns_answer *answer;
663     LPSTR CurrentName;
664     unsigned CNameLoop;
665     PFIXED_INFO network_info;
666     ULONG network_info_blen = 0;
667     DWORD network_info_result;
668     PIP_ADDR_STRING pip;
669     IP4_ADDRESS Address;
670     struct in_addr addr;
671     PCHAR HostWithDomainName;
672     PCHAR AnsiName;
673     size_t NameLen = 0;
674 
675     if (Name == NULL)
676         return ERROR_INVALID_PARAMETER;
677     if (QueryResultSet == NULL)
678         return ERROR_INVALID_PARAMETER;
679 
680     *QueryResultSet = NULL;
681 
682     switch (Type)
683     {
684     case DNS_TYPE_A:
685         /* FIXME: how much instead of MAX_PATH? */
686         NameLen = WideCharToMultiByte(CP_ACP,
687                                       0,
688                                       Name,
689                                       -1,
690                                       NULL,
691                                       0,
692                                       NULL,
693                                       0);
694         AnsiName = RtlAllocateHeap(RtlGetProcessHeap(), 0, NameLen);
695         if (NULL == AnsiName)
696         {
697             return ERROR_OUTOFMEMORY;
698         }
699         WideCharToMultiByte(CP_ACP,
700                             0,
701                             Name,
702                             -1,
703                             AnsiName,
704                             NameLen,
705                             NULL,
706                             0);
707         NameLen--;
708 
709         network_info_result = GetNetworkParams(NULL, &network_info_blen);
710         network_info = (PFIXED_INFO)RtlAllocateHeap(RtlGetProcessHeap(), 0, (size_t)network_info_blen);
711         if (NULL == network_info)
712         {
713             return ERROR_OUTOFMEMORY;
714         }
715 
716         network_info_result = GetNetworkParams(network_info, &network_info_blen);
717         if (network_info_result != ERROR_SUCCESS)
718         {
719             RtlFreeHeap(RtlGetProcessHeap(), 0, network_info);
720             return network_info_result;
721         }
722 
723         if ((Address = CheckForCurrentHostname(NameLen != 0 ? AnsiName : network_info->HostName, network_info)) != 0)
724         {
725             size_t TempLen = 2, StringLength = 0;
726             RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
727             StringCchLengthA(network_info->HostName, sizeof(network_info->HostName), &StringLength);
728             TempLen += StringLength;
729             StringCchLengthA(network_info->DomainName, sizeof(network_info->DomainName), &StringLength);
730             TempLen += StringLength;
731             HostWithDomainName = (PCHAR)RtlAllocateHeap(RtlGetProcessHeap(), 0, TempLen);
732             StringCchCopyA(HostWithDomainName, TempLen, network_info->HostName);
733             if (network_info->DomainName[0])
734             {
735                 StringCchCatA(HostWithDomainName, TempLen, ".");
736                 StringCchCatA(HostWithDomainName, TempLen, network_info->DomainName);
737             }
738             RtlFreeHeap(RtlGetProcessHeap(), 0, network_info);
739             *QueryResultSet = (PDNS_RECORD)RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_RECORD));
740 
741             if (NULL == *QueryResultSet)
742             {
743                 RtlFreeHeap(RtlGetProcessHeap(), 0, HostWithDomainName);
744                 return ERROR_OUTOFMEMORY;
745             }
746 
747             (*QueryResultSet)->pNext = NULL;
748             (*QueryResultSet)->wType = Type;
749             (*QueryResultSet)->wDataLength = sizeof(DNS_A_DATA);
750             (*QueryResultSet)->Flags.S.Section = DnsSectionAnswer;
751             (*QueryResultSet)->Flags.S.CharSet = DnsCharSetUnicode;
752             (*QueryResultSet)->Data.A.IpAddress = Address;
753 
754             (*QueryResultSet)->pName = (LPSTR)DnsCToW(HostWithDomainName);
755 
756             RtlFreeHeap(RtlGetProcessHeap(), 0, HostWithDomainName);
757             return (*QueryResultSet)->pName ? ERROR_SUCCESS : ERROR_OUTOFMEMORY;
758         }
759 
760         if ((Options & DNS_QUERY_NO_WIRE_QUERY) != 0)
761         {
762             RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
763             RtlFreeHeap(RtlGetProcessHeap(), 0, network_info);
764             return ERROR_FILE_NOT_FOUND;
765         }
766 
767         adns_error = adns_init(&astate, adns_if_noenv | adns_if_noerrprint | adns_if_noserverwarn, 0);
768         if (adns_error != adns_s_ok)
769         {
770             RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
771             RtlFreeHeap(RtlGetProcessHeap(), 0, network_info);
772             return DnsIntTranslateAdnsToDNS_STATUS(adns_error);
773         }
774         for (pip = &(network_info->DnsServerList); pip; pip = pip->Next)
775         {
776             addr.s_addr = inet_addr(pip->IpAddress.String);
777             if ((addr.s_addr != INADDR_ANY) && (addr.s_addr != INADDR_NONE))
778                 adns_addserver(astate, addr);
779         }
780         if (network_info->DomainName[0])
781         {
782             adns_ccf_search(astate, "LOCALDOMAIN", -1, network_info->DomainName);
783         }
784         RtlFreeHeap(RtlGetProcessHeap(), 0, network_info);
785 
786         if (!adns_numservers(astate))
787         {
788             /* There are no servers to query so bail out */
789             adns_finish(astate);
790             RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
791             return ERROR_FILE_NOT_FOUND;
792         }
793 
794         /*
795         * adns doesn't resolve chained CNAME records (a CNAME which points to
796         * another CNAME pointing to another... pointing to an A record), according
797         * to a mailing list thread the authors believe that chained CNAME records
798         * are invalid and the DNS entries should be fixed. That's a nice academic
799         * standpoint, but there certainly are chained CNAME records out there,
800         * even some fairly major ones (at the time of this writing
801         * download.mozilla.org is a chained CNAME). Everyone else seems to resolve
802         * these fine, so we should too. So we loop here to try to resolve CNAME
803         * chains ourselves. Of course, there must be a limit to protect against
804         * CNAME loops.
805         */
806 
807 #define CNAME_LOOP_MAX 16
808 
809         CurrentName = AnsiName;
810 
811         for (CNameLoop = 0; CNameLoop < CNAME_LOOP_MAX; CNameLoop++)
812         {
813             adns_error = adns_synchronous(astate, CurrentName, adns_r_addr, quflags, &answer);
814 
815             if (adns_error != adns_s_ok)
816             {
817                 adns_finish(astate);
818 
819                 if (CurrentName != AnsiName)
820                     RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName);
821 
822                 RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
823                 return DnsIntTranslateAdnsToDNS_STATUS(adns_error);
824             }
825 
826             if (answer && answer->rrs.addr)
827             {
828                 if (CurrentName != AnsiName)
829                     RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName);
830 
831                 RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
832                 *QueryResultSet = (PDNS_RECORD)RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(DNS_RECORD));
833 
834                 if (NULL == *QueryResultSet)
835                 {
836                     adns_finish(astate);
837                     return ERROR_OUTOFMEMORY;
838                 }
839 
840                 (*QueryResultSet)->pNext = NULL;
841                 (*QueryResultSet)->wType = Type;
842                 (*QueryResultSet)->wDataLength = sizeof(DNS_A_DATA);
843                 (*QueryResultSet)->Flags.S.Section = DnsSectionAnswer;
844                 (*QueryResultSet)->Flags.S.CharSet = DnsCharSetUnicode;
845                 (*QueryResultSet)->Data.A.IpAddress = answer->rrs.addr->addr.inet.sin_addr.s_addr;
846 
847                 adns_finish(astate);
848 
849                 (*QueryResultSet)->pName = (LPSTR)xstrsave(Name);
850 
851                 return (*QueryResultSet)->pName ? ERROR_SUCCESS : ERROR_OUTOFMEMORY;
852             }
853 
854             if (NULL == answer || adns_s_prohibitedcname != answer->status || NULL == answer->cname)
855             {
856                 adns_finish(astate);
857 
858                 if (CurrentName != AnsiName)
859                     RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName);
860 
861                 RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
862                 return ERROR_FILE_NOT_FOUND;
863             }
864 
865             if (CurrentName != AnsiName)
866                 RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName);
867 
868             CurrentName = (LPSTR)xstrsaveA(answer->cname);
869 
870             if (!CurrentName)
871             {
872                 RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
873                 adns_finish(astate);
874                 return ERROR_OUTOFMEMORY;
875             }
876         }
877 
878         adns_finish(astate);
879         RtlFreeHeap(RtlGetProcessHeap(), 0, AnsiName);
880         RtlFreeHeap(RtlGetProcessHeap(), 0, CurrentName);
881         return ERROR_FILE_NOT_FOUND;
882 
883     default:
884         return ERROR_OUTOFMEMORY; /* XXX arty: find a better error code. */
885     }
886 }
887 
888 void
889 DnsIntFreeRecordList(PDNS_RECORD ToDelete)
890 {
891     UINT i;
892     PDNS_RECORD next = 0;
893 
894     while(ToDelete)
895     {
896         if(ToDelete->pName)
897             RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete->pName);
898 
899         switch(ToDelete->wType)
900         {
901             case DNS_TYPE_CNAME:
902             case DNS_TYPE_PTR:
903             case DNS_TYPE_NS:
904             case DNS_TYPE_MB:
905             case DNS_TYPE_MD:
906             case DNS_TYPE_MF:
907             case DNS_TYPE_MG:
908             case DNS_TYPE_MR:
909                 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete->Data.PTR.pNameHost);
910                 break;
911 
912             case DNS_TYPE_MINFO:
913             case DNS_TYPE_MX:
914                 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete->Data.MX.pNameExchange);
915                 break;
916 
917             case DNS_TYPE_HINFO:
918                 for(i = 0; i < ToDelete->Data.TXT.dwStringCount; i++)
919                     RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete->Data.TXT.pStringArray[i]);
920 
921                 RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete->Data.TXT.pStringArray);
922                 break;
923         }
924 
925         next = ToDelete->pNext;
926         RtlFreeHeap(RtlGetProcessHeap(), 0, ToDelete);
927         ToDelete = next;
928     }
929 }
930 
931 BOOL
932 WINAPI
933 DnsFlushResolverCache(VOID)
934 {
935     DNS_STATUS Status = ERROR_SUCCESS;
936 
937     DPRINT("DnsFlushResolverCache()\n");
938 
939     RpcTryExcept
940     {
941         Status = R_ResolverFlushCache(NULL);
942         DPRINT("R_ResolverFlushCache() returned %lu\n", Status);
943     }
944     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
945     {
946         Status = RpcExceptionCode();
947         DPRINT("Exception returned %lu\n", Status);
948     }
949     RpcEndExcept;
950 
951     return (Status == ERROR_SUCCESS);
952 }
953 
954 BOOL
955 WINAPI
956 DnsGetCacheDataTable(
957     _Out_ PDNS_CACHE_ENTRY *DnsCache)
958 {
959     DNS_STATUS Status = ERROR_SUCCESS;
960     PDNS_CACHE_ENTRY CacheEntries = NULL;
961 
962     if (DnsCache == NULL)
963         return FALSE;
964 
965     RpcTryExcept
966     {
967         Status = CRrReadCache(NULL,
968                               &CacheEntries);
969         DPRINT("CRrReadCache() returned %lu\n", Status);
970     }
971     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
972     {
973         Status = RpcExceptionCode();
974         DPRINT1("Exception returned %lu\n", Status);
975     }
976     RpcEndExcept;
977 
978     if (Status != ERROR_SUCCESS)
979         return FALSE;
980 
981     if (CacheEntries == NULL)
982         return FALSE;
983 
984     *DnsCache = CacheEntries;
985 
986     return TRUE;
987 }
988 
989 DWORD
990 WINAPI
991 GetCurrentTimeInSeconds(VOID)
992 {
993     FILETIME Time;
994     FILETIME Adjustment;
995     ULARGE_INTEGER lTime, lAdj;
996     SYSTEMTIME st = {1970, 1, 0, 1, 0, 0, 0};
997 
998     SystemTimeToFileTime(&st, &Adjustment);
999     memcpy(&lAdj, &Adjustment, sizeof(lAdj));
1000     GetSystemTimeAsFileTime(&Time);
1001     memcpy(&lTime, &Time, sizeof(lTime));
1002     lTime.QuadPart -= lAdj.QuadPart;
1003     return (DWORD)(lTime.QuadPart/10000000ULL);
1004 }
1005