liblog_SystemConfiguration.m   [plain text]


/*
 * Copyright (c) 2017 Apple Inc. All rights reserved.
 *
 * @APPLE_LICENSE_HEADER_START@
 *
 * This file contains Original Code and/or Modifications of Original Code
 * as defined in and that are subject to the Apple Public Source License
 * Version 2.0 (the 'License'). You may not use this file except in
 * compliance with the License. Please obtain a copy of the License at
 * http://www.opensource.apple.com/apsl/ and read it before using this
 * file.
 *
 * The Original Code and all software distributed under the License are
 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
 * Please see the License for the specific language governing rights and
 * limitations under the License.
 *
 * @APPLE_LICENSE_HEADER_END@
 */

/*
 * Modification History
 *
 * May 18, 2017		Allan Nathanson <ajn@apple.com>
 * - initial revision
 */

#define	_LIBLOG_SYSTEMCONFIGURATION_

#import <Foundation/Foundation.h>
#import <os/log_private.h>
#import <os/state_private.h>
#import <string.h>

#import <dnsinfo.h>
#import "dnsinfo_internal.h"
#import <network_information.h>

#define	my_log(__level, __format, ...)	[string appendFormat:@(__format "\n"), ## __VA_ARGS__]
#define my_log_context_type	NSMutableString *
#define	my_log_context_name	string
#import "dnsinfo_logging.h"
#import "network_state_information_logging.h"
#undef	my_log_context_name
#undef	my_log_context_type
#undef	my_log

#define SCAS(str)			\
	[[NSAttributedString alloc] initWithString:str]

#define SCASWithFormat(format, ...)	\
	SCAS(([[NSString alloc] initWithFormat:format, ##__VA_ARGS__]))

#pragma mark -
#pragma mark os_log formatting entry point

struct SC_OSLog_Formatters {
	const char *type;
	NS_RETURNS_RETAINED NSAttributedString * (*function)(id value);
};

NS_RETURNS_RETAINED
NSAttributedString *
OSLogCopyFormattedString(const char *type, id value, os_log_type_info_t info)
{
#pragma	unused(info)
	// add functions for each type into this list
	static const struct SC_OSLog_Formatters formatters[] = {
//		{ .type = "???",  .function = _SC_OSLogCopyFormattedString_??? },
	};

	for (int i = 0; i < (int)(sizeof(formatters) / sizeof(formatters[0])); i++) {
		if (strcmp(type, formatters[i].type) == 0) {
			return formatters[i].function(value);
		}
	}

	return SCASWithFormat(@"liblog_SystemConfiguration: Not yet supported os_log formatting type: %s", type);
}

#pragma mark -
#pragma mark os_state formatting entry point

#define SCNS(str)			\
	[[NSString alloc] initWithString:(str)]

#define SCNSWithFormat(format, ...)			\
	[[NSString alloc] initWithFormat:format, ##__VA_ARGS__]

static NS_RETURNS_RETAINED NSString *
_SC_OSStateCopyFormattedString_dnsinfo(uint32_t data_size, void *data)
{
	dns_config_t		*dns_config	= NULL;
	_dns_config_buf_t	*dns_config_buf;
	NSMutableString		*string;

	// os_state_add_handler w/
	//	osd_type                 = OS_STATE_DATA_CUSTOM
	//	osd_decoder.osdd_library = "SystemConfiguration
	//	osd_decoder.osdd_type    = "dnsinfo"

	if ((data_size == 0) || (data == NULL)) {
		return @"No DNS configuration";
	} else if (data_size < sizeof(_dns_config_buf_t)) {
		return SCNSWithFormat(@"DNS configuration: size error (%d < %zd)",
				      data_size,
				      sizeof(_dns_config_buf_t));
	}

	dns_config_buf = _dns_configuration_buffer_create(data, data_size);
	if (dns_config_buf == NULL) {
		return @"DNS configuration: data error";
	}

	dns_config = _dns_configuration_buffer_expand(dns_config_buf);
	if (dns_config == NULL) {
		// if we were unable to expand the configuration
		_dns_configuration_buffer_free(&dns_config_buf);
		return @"DNS configuration: expansion error";
	}

	string = [NSMutableString string];
	_dns_configuration_log(dns_config, TRUE, string);
	if (string.length == 0) {
		[string appendString:@"DNS configuration: not available"];
	}
	free(dns_config);

	return string;
}

static NS_RETURNS_RETAINED NSString *
_SC_OSStateCopyFormattedString_nwi(uint32_t data_size, void *data)
{
	nwi_state_t		state	= (nwi_state_t)data;
	NSMutableString		*string;

	// os_state_add_handler w/
	//	osd_type                 = OS_STATE_DATA_CUSTOM
	//	osd_decoder.osdd_library = "SystemConfiguration
	//	osd_decoder.osdd_type    = "nwi"

	if ((data_size == 0) || (data == NULL)) {
		return @"No network information";
	} else if (data_size < sizeof(nwi_state)) {
		return SCNSWithFormat(@"Network information: size error (%d < %zd)",
				      data_size,
				      sizeof(_dns_config_buf_t));
	} else if (state->version != NWI_STATE_VERSION) {
		return SCNSWithFormat(@"Network information: version error (%d != %d)",
				      state->version,
				      NWI_STATE_VERSION);
	}

	string = [NSMutableString string];
	_nwi_state_log(state, TRUE, string);
	if (string.length == 0) {
		[string appendString:@"Network information: not available"];
	}

	return string;
}

struct SC_OSState_Formatters {
	const char *type;
	NS_RETURNS_RETAINED NSString * (*function)(uint32_t data_size, void *data);
};

NS_RETURNS_RETAINED
NSString *
OSStateCreateStringWithData(const char *type, uint32_t data_size, void *data)
{
	// add functions for each type into this list
	static const struct SC_OSState_Formatters formatters[] = {
		{ .type = "dnsinfo",	.function = _SC_OSStateCopyFormattedString_dnsinfo	},
		{ .type = "nwi",	.function = _SC_OSStateCopyFormattedString_nwi		},
	};

	for (int i = 0; i < (int)(sizeof(formatters) / sizeof(formatters[0])); i++) {
		if (strcmp(type, formatters[i].type) == 0) {
			return formatters[i].function(data_size, data);
		}
	}

	return SCNSWithFormat(@"liblog_SystemConfiguration: Not yet supported os_state formatting type: %s", type);
}