util/dnsconfig: add new package to parse macOS DNS configuration
Signed-off-by: Andrew Dunham <andrew@du.nham.ca> Change-Id: I271b401af1b5e626b3afe291c6f7f15b319c601dandrew/util-dnsconfig
parent
e1530cdfcc
commit
9058121b0c
|
@ -0,0 +1,217 @@
|
|||
package dnsconfig
|
||||
|
||||
/*
|
||||
#cgo LDFLAGS: -ldl
|
||||
|
||||
#include <dlfcn.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
void* call_pointer(void* addr) {
|
||||
void* (*fn)(void) = addr;
|
||||
return fn();
|
||||
}
|
||||
|
||||
void call_arg(void* addr, void* arg) {
|
||||
void (*fn)(void*) = addr;
|
||||
fn(arg);
|
||||
}
|
||||
*/
|
||||
import "C"
|
||||
|
||||
import (
|
||||
"encoding/binary"
|
||||
"errors"
|
||||
"fmt"
|
||||
"net/netip"
|
||||
"sync"
|
||||
"syscall"
|
||||
"unsafe"
|
||||
)
|
||||
|
||||
var (
|
||||
fptrOnce sync.Once
|
||||
dnsConfigurationCopyPtr unsafe.Pointer
|
||||
dnsConfigurationFreePtr unsafe.Pointer
|
||||
)
|
||||
|
||||
func initPointers() {
|
||||
fptrOnce.Do(func() {
|
||||
sym := C.CString("dns_configuration_copy")
|
||||
defer C.free(unsafe.Pointer(sym))
|
||||
dnsConfigurationCopyPtr = C.dlsym(C.RTLD_DEFAULT, sym)
|
||||
|
||||
sym = C.CString("dns_configuration_free")
|
||||
defer C.free(unsafe.Pointer(sym))
|
||||
dnsConfigurationFreePtr = C.dlsym(C.RTLD_DEFAULT, sym)
|
||||
})
|
||||
}
|
||||
|
||||
var errSymbolNotFound = errors.New("symbol not found")
|
||||
|
||||
func dnsConfigurationCopy() (*dnsConfig, error) {
|
||||
initPointers()
|
||||
if dnsConfigurationCopyPtr == nil {
|
||||
return nil, errSymbolNotFound
|
||||
}
|
||||
|
||||
// Call through cgo so that the Go runtime switches to a C stack.
|
||||
ptr := C.call_pointer(dnsConfigurationCopyPtr)
|
||||
return (*dnsConfig)(ptr), nil
|
||||
}
|
||||
|
||||
func dnsConfigurationFree(p *dnsConfig) error {
|
||||
initPointers()
|
||||
if dnsConfigurationFreePtr == nil {
|
||||
return errSymbolNotFound
|
||||
}
|
||||
// Call through cgo so that the Go runtime switches to a C stack.
|
||||
C.call_arg(dnsConfigurationFreePtr, unsafe.Pointer(p))
|
||||
return nil
|
||||
}
|
||||
|
||||
// DNSConfig contains DNS configuration information as returned by macOS. It is
|
||||
// the Go version of the private dns_config_t type.
|
||||
type DNSConfig struct {
|
||||
Resolvers []*DNSResolver
|
||||
ScopedResolvers []*DNSResolver
|
||||
Generation uint64
|
||||
ServiceSpecificResolvers []*DNSResolver
|
||||
Version uint32
|
||||
}
|
||||
|
||||
// DNSResolver contains DNS resolver-specific information as returned by macOS.
|
||||
// It is the Go version of the private dns_resolver_t type.
|
||||
type DNSResolver struct {
|
||||
Domain string
|
||||
Nameservers []netip.AddrPort
|
||||
Port uint16
|
||||
Search []string
|
||||
Options string
|
||||
Timeout uint32
|
||||
SearchOrder uint32
|
||||
IfIndex uint32
|
||||
Flags uint32
|
||||
ReachFlags uint32
|
||||
ServiceIdentifier uint32
|
||||
CID string
|
||||
IfName string
|
||||
|
||||
// TODO: SortAddr []any?
|
||||
}
|
||||
|
||||
// Get returns this system's DNS configuration, or an error.
|
||||
func Get() (*DNSConfig, error) {
|
||||
config, err := dnsConfigurationCopy()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer dnsConfigurationFree(config)
|
||||
|
||||
// Verify that the version is what we expect. On newer versions of
|
||||
// macOS, we could check this and only load fields that are present,
|
||||
// instead of failing outright.
|
||||
version := binary.LittleEndian.Uint32(config.data[44 : 44+4])
|
||||
if version != 20170629 {
|
||||
return nil, fmt.Errorf("version mismatch: %d != 20170629", version)
|
||||
}
|
||||
|
||||
ret := &DNSConfig{
|
||||
Generation: binary.LittleEndian.Uint64(config.data[24 : 24+8]),
|
||||
Version: version,
|
||||
}
|
||||
|
||||
// Populate resolvers
|
||||
for _, resolver := range getResolvers(config.data[:], 0, 4) {
|
||||
ret.Resolvers = append(ret.Resolvers, parseResolver(resolver))
|
||||
}
|
||||
for _, resolver := range getResolvers(config.data[:], 12, 16) {
|
||||
ret.ScopedResolvers = append(ret.ScopedResolvers, parseResolver(resolver))
|
||||
}
|
||||
for _, resolver := range getResolvers(config.data[:], 32, 36) {
|
||||
ret.ServiceSpecificResolvers = append(ret.ServiceSpecificResolvers, parseResolver(resolver))
|
||||
}
|
||||
|
||||
return ret, nil
|
||||
}
|
||||
|
||||
func getResolvers(data []byte, numOff, arrOff int) []*dnsResolver {
|
||||
n := int(binary.LittleEndian.Uint32(data[numOff : numOff+4]))
|
||||
arr := unsafe.Pointer(uintptr(binary.LittleEndian.Uint64(data[arrOff : arrOff+8])))
|
||||
return unsafe.Slice((**dnsResolver)(arr), n)
|
||||
}
|
||||
|
||||
func parseResolver(r *dnsResolver) *DNSResolver {
|
||||
ret := &DNSResolver{
|
||||
Domain: r.readCharPtr(0),
|
||||
Port: binary.LittleEndian.Uint16(r.data[20 : 20+2]),
|
||||
Options: r.readCharPtr(48),
|
||||
Timeout: r.readUint32(56),
|
||||
SearchOrder: r.readUint32(60),
|
||||
IfIndex: r.readUint32(64),
|
||||
Flags: r.readUint32(68),
|
||||
ReachFlags: r.readUint32(72),
|
||||
ServiceIdentifier: r.readUint32(76),
|
||||
CID: r.readCharPtr(80),
|
||||
IfName: r.readCharPtr(88),
|
||||
}
|
||||
|
||||
// The actual nameservers for this DNS entry.
|
||||
nNameservers := int(binary.LittleEndian.Uint32(r.data[8 : 8+4]))
|
||||
arr := unsafe.Pointer(uintptr(binary.LittleEndian.Uint64(r.data[12 : 12+8])))
|
||||
for _, sockaddr := range unsafe.Slice((**syscall.RawSockaddr)(arr), nNameservers) {
|
||||
switch sockaddr.Family {
|
||||
case syscall.AF_INET:
|
||||
sa := (*syscall.RawSockaddrInet4)(unsafe.Pointer(sockaddr))
|
||||
ret.Nameservers = append(ret.Nameservers, netip.AddrPortFrom(
|
||||
netip.AddrFrom4(sa.Addr),
|
||||
sa.Port,
|
||||
))
|
||||
|
||||
case syscall.AF_INET6:
|
||||
sa := (*syscall.RawSockaddrInet6)(unsafe.Pointer(sockaddr))
|
||||
ret.Nameservers = append(ret.Nameservers, netip.AddrPortFrom(
|
||||
netip.AddrFrom16(sa.Addr),
|
||||
sa.Port,
|
||||
))
|
||||
|
||||
default:
|
||||
// Skip unknown address families
|
||||
// TODO: log?
|
||||
}
|
||||
}
|
||||
|
||||
// Search domains
|
||||
nSearch := int(binary.LittleEndian.Uint32(r.data[24 : 24+4]))
|
||||
arr = unsafe.Pointer(uintptr(binary.LittleEndian.Uint64(r.data[28 : 28+8])))
|
||||
for _, ss := range unsafe.Slice((**C.char)(arr), nSearch) {
|
||||
ret.Search = append(ret.Search, C.GoString(ss))
|
||||
}
|
||||
|
||||
return ret
|
||||
}
|
||||
|
||||
// dnsConfig is the type returned from the dns_configuration_copy function. The
|
||||
// C header sets #pragma pack(4), which isn't easily represented in Go; we
|
||||
// instead use binary.Read to get fields from this structure.
|
||||
type dnsConfig struct {
|
||||
data [48]byte
|
||||
}
|
||||
|
||||
// dnsResolver is the dns_resolver_t type; as above, since we can't represent
|
||||
// it in Go, we read fields from the structure manually.
|
||||
type dnsResolver struct {
|
||||
data [96]byte
|
||||
}
|
||||
|
||||
func (d *dnsResolver) readCharPtr(off int) string {
|
||||
ptr := unsafe.Pointer(uintptr(binary.LittleEndian.Uint64(d.data[off : off+8])))
|
||||
return C.GoString((*C.char)(ptr))
|
||||
}
|
||||
|
||||
func (d *dnsResolver) readInt32(off int) int32 {
|
||||
return int32(binary.LittleEndian.Uint32(d.data[off : off+4]))
|
||||
}
|
||||
|
||||
func (d *dnsResolver) readUint32(off int) uint32 {
|
||||
return binary.LittleEndian.Uint32(d.data[off : off+4])
|
||||
}
|
|
@ -0,0 +1,30 @@
|
|||
package dnsconfig
|
||||
|
||||
import "testing"
|
||||
|
||||
func TestGet(t *testing.T) {
|
||||
config, err := Get()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
if len(config.Resolvers) < 1 {
|
||||
t.Fatal("wanted at least one resolver")
|
||||
}
|
||||
|
||||
// Sensibility check: do we have at least one nameserver?
|
||||
var nameservers int
|
||||
for _, resolver := range config.Resolvers {
|
||||
nameservers += len(resolver.Nameservers)
|
||||
}
|
||||
for _, resolver := range config.ScopedResolvers {
|
||||
nameservers += len(resolver.Nameservers)
|
||||
}
|
||||
for _, resolver := range config.ServiceSpecificResolvers {
|
||||
nameservers += len(resolver.Nameservers)
|
||||
}
|
||||
|
||||
if nameservers == 0 {
|
||||
t.Fatal("wanted at least one nameserver, got 0")
|
||||
}
|
||||
}
|
|
@ -0,0 +1,114 @@
|
|||
#include <sys/cdefs.h>
|
||||
#include <stddef.h>
|
||||
#include <stdint.h>
|
||||
#include <stdio.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/socket.h>
|
||||
#include <netinet/in.h>
|
||||
|
||||
/**************************************************
|
||||
* BEGIN TYPES FROM APPLE HEADER
|
||||
**************************************************/
|
||||
|
||||
#define DNS_PTR(type, name) \
|
||||
union { \
|
||||
type name; \
|
||||
uint64_t _ ## name ## _p; \
|
||||
}
|
||||
|
||||
#define DNS_VAR(type, name) \
|
||||
type name
|
||||
|
||||
|
||||
#pragma pack(4)
|
||||
typedef struct {
|
||||
struct in_addr address;
|
||||
struct in_addr mask;
|
||||
} dns_sortaddr_t;
|
||||
#pragma pack()
|
||||
|
||||
|
||||
#pragma pack(4)
|
||||
typedef struct {
|
||||
DNS_PTR(char *, domain); /* domain */
|
||||
DNS_VAR(int32_t, n_nameserver); /* # nameserver */
|
||||
DNS_PTR(struct sockaddr **, nameserver);
|
||||
DNS_VAR(uint16_t, port); /* port (in host byte order) */
|
||||
DNS_VAR(int32_t, n_search); /* # search */
|
||||
DNS_PTR(char **, search);
|
||||
DNS_VAR(int32_t, n_sortaddr); /* # sortaddr */
|
||||
DNS_PTR(dns_sortaddr_t **, sortaddr);
|
||||
DNS_PTR(char *, options); /* options */
|
||||
DNS_VAR(uint32_t, timeout); /* timeout */
|
||||
DNS_VAR(uint32_t, search_order); /* search_order */
|
||||
DNS_VAR(uint32_t, if_index);
|
||||
DNS_VAR(uint32_t, flags);
|
||||
DNS_VAR(uint32_t, reach_flags); /* SCNetworkReachabilityFlags */
|
||||
DNS_VAR(uint32_t, service_identifier);
|
||||
DNS_PTR(char *, cid); /* configuration identifer */
|
||||
DNS_PTR(char *, if_name); /* if_index interface name */
|
||||
} dns_resolver_t;
|
||||
#pragma pack()
|
||||
|
||||
#pragma pack(4)
|
||||
typedef struct {
|
||||
DNS_VAR(int32_t, n_resolver); /* resolver configurations */
|
||||
DNS_PTR(dns_resolver_t **, resolver);
|
||||
DNS_VAR(int32_t, n_scoped_resolver); /* "scoped" resolver configurations */
|
||||
DNS_PTR(dns_resolver_t **, scoped_resolver);
|
||||
DNS_VAR(uint64_t, generation);
|
||||
DNS_VAR(int32_t, n_service_specific_resolver);
|
||||
DNS_PTR(dns_resolver_t **, service_specific_resolver);
|
||||
DNS_VAR(uint32_t, version);
|
||||
} dns_config_t;
|
||||
#pragma pack()
|
||||
|
||||
/**************************************************
|
||||
* END TYPES FROM APPLE HEADER
|
||||
**************************************************/
|
||||
|
||||
#define field_info(type, field) \
|
||||
printf("%-15s\t%-30s\toffset=%lu\tsizeof=%lu\n", \
|
||||
#type, \
|
||||
#field, \
|
||||
offsetof(type, field) , \
|
||||
sizeof ((type *)0)->field \
|
||||
)
|
||||
|
||||
int main(void) {
|
||||
printf("sizeof(dns_config_t)=%lu\n", sizeof(dns_config_t));
|
||||
field_info(dns_config_t, n_resolver);
|
||||
field_info(dns_config_t, resolver);
|
||||
field_info(dns_config_t, n_scoped_resolver);
|
||||
field_info(dns_config_t, scoped_resolver);
|
||||
field_info(dns_config_t, generation);
|
||||
field_info(dns_config_t, n_service_specific_resolver);
|
||||
field_info(dns_config_t, service_specific_resolver);
|
||||
field_info(dns_config_t, version);
|
||||
|
||||
printf("\n");
|
||||
|
||||
printf("sizeof(dns_resolver_t)=%lu\n", sizeof(dns_resolver_t));
|
||||
field_info(dns_resolver_t, domain);
|
||||
field_info(dns_resolver_t, n_nameserver);
|
||||
field_info(dns_resolver_t, nameserver);
|
||||
field_info(dns_resolver_t, port);
|
||||
field_info(dns_resolver_t, n_search);
|
||||
field_info(dns_resolver_t, search);
|
||||
field_info(dns_resolver_t, n_sortaddr);
|
||||
field_info(dns_resolver_t, sortaddr);
|
||||
field_info(dns_resolver_t, options);
|
||||
field_info(dns_resolver_t, timeout);
|
||||
field_info(dns_resolver_t, search_order);
|
||||
field_info(dns_resolver_t, if_index);
|
||||
field_info(dns_resolver_t, flags);
|
||||
field_info(dns_resolver_t, reach_flags);
|
||||
field_info(dns_resolver_t, service_identifier);
|
||||
field_info(dns_resolver_t, cid);
|
||||
field_info(dns_resolver_t, if_name);
|
||||
|
||||
printf("\n");
|
||||
field_info(struct sockaddr, sa_len);
|
||||
field_info(struct sockaddr, sa_family);
|
||||
field_info(struct sockaddr, sa_data);
|
||||
}
|
Loading…
Reference in New Issue