/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License, Version 1.0 only
* (the "License"). You may not use this file except in compliance
* with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/*
* Copyright (c) 2002-2003, Network Appliance, Inc. All rights reserved.
*/
/*
* Copyright 2004 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
#pragma ident "%Z%%M% %I% %E% SMI"
/*
*
* MODULE: dat_sr_parser.c
*
* PURPOSE: static registry parser
*
* $Id: udat_sr_parser.c,v 1.1 2003/07/31 14:04:19 jlentini Exp $
*/
#include "udat_sr_parser.h"
#include "dat_sr.h"
/*
*
* Constants
*
*/
#define DAT_SR_CONF_ENV "DAT_OVERRIDE"
#define DAT_SR_CONF_DEFAULT "/etc/dat/dat.conf"
#define DAT_SR_TOKEN_THREADSAFE "threadsafe"
#define DAT_SR_TOKEN_NONTHREADSAFE "nonthreadsafe"
#define DAT_SR_TOKEN_DEFAULT "default"
#define DAT_SR_TOKEN_NONDEFAULT "nondefault"
#define DAT_SR_CHAR_NEWLINE '\n'
#define DAT_SR_CHAR_COMMENT '#'
#define DAT_SR_CHAR_QUOTE '"'
#define DAT_SR_CHAR_BACKSLASH '\\'
/*
*
* Enumerations
*
*/
typedef enum
{
DAT_SR_TOKEN_STRING, /* text field (both quoted or unquoted) */
DAT_SR_TOKEN_EOR, /* end of record (newline) */
DAT_SR_TOKEN_EOF /* end of file */
} DAT_SR_TOKEN_TYPE;
typedef enum
{
DAT_SR_API_UDAT,
DAT_SR_API_KDAT
} DAT_SR_API_TYPE;
/*
*
* Structures
*
*/
typedef struct
{
DAT_SR_TOKEN_TYPE type;
char *value; /* valid if type is DAT_SR_TOKEN_STRING */
DAT_OS_SIZE value_len;
} DAT_SR_TOKEN;
typedef struct DAT_SR_STACK_NODE
{
DAT_SR_TOKEN token;
struct DAT_SR_STACK_NODE *next;
} DAT_SR_STACK_NODE;
typedef struct
{
DAT_UINT32 major;
DAT_UINT32 minor;
} DAT_SR_VERSION;
typedef struct
{
char *id;
DAT_SR_VERSION version;
} DAT_SR_PROVIDER_VERSION;
typedef struct
{
DAT_SR_API_TYPE type;
DAT_SR_VERSION version;
} DAT_SR_API_VERSION;
typedef struct
{
char *ia_name;
DAT_SR_API_VERSION api_version;
DAT_BOOLEAN is_thread_safe;
DAT_BOOLEAN is_default;
char *lib_path;
DAT_SR_PROVIDER_VERSION provider_version;
char *ia_params;
char *platform_params;
} DAT_SR_CONF_ENTRY;
/*
*
* Internal Function Declarations
*
*/
static DAT_RETURN
dat_sr_load_entry(
DAT_SR_CONF_ENTRY *entry);
static DAT_BOOLEAN
dat_sr_is_valid_entry(
DAT_SR_CONF_ENTRY *entry);
static char *
dat_sr_type_to_str(
DAT_SR_TOKEN_TYPE type);
static DAT_RETURN
dat_sr_parse_eof(
DAT_OS_FILE *file);
static DAT_RETURN
dat_sr_parse_entry(
DAT_OS_FILE *file);
static DAT_RETURN
dat_sr_parse_ia_name(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_parse_api(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_parse_thread_safety(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_parse_default(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_parse_lib_path(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_parse_provider_version(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_parse_ia_params(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_parse_platform_params(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_parse_eoe(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry);
static DAT_RETURN
dat_sr_convert_api(
char *str,
DAT_SR_API_VERSION *api_version);
static DAT_RETURN
dat_sr_convert_thread_safety(
char *str,
DAT_BOOLEAN *is_thread_safe);
static DAT_RETURN
dat_sr_convert_default(
char *str,
DAT_BOOLEAN *is_default);
static DAT_RETURN
dat_sr_convert_provider_version(
char *str,
DAT_SR_PROVIDER_VERSION *provider_version);
static DAT_RETURN
dat_sr_get_token(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token);
static DAT_RETURN
dat_sr_put_token(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token);
static DAT_RETURN
dat_sr_read_token(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token);
static DAT_RETURN
dat_sr_read_str(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token,
DAT_OS_SIZE token_len);
static DAT_RETURN
dat_sr_read_quoted_str(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token,
DAT_OS_SIZE token_len,
DAT_COUNT num_escape_seq);
static void
dat_sr_read_comment(
DAT_OS_FILE *file);
/*
*
* Global Variables
*
*/
static DAT_SR_STACK_NODE *g_token_stack = NULL;
/*
*
* External Function Definitions
*
*/
/*
* Function: dat_sr_load
*/
DAT_RETURN
dat_sr_load(void)
{
char *sr_path;
DAT_OS_FILE *sr_file;
sr_path = dat_os_getenv(DAT_SR_CONF_ENV);
if (sr_path == NULL) {
sr_path = DAT_SR_CONF_DEFAULT;
}
dat_os_dbg_print(DAT_OS_DBG_TYPE_SR,
"DAT Registry: static registry file <%s> \n", sr_path);
sr_file = dat_os_fopen(sr_path);
if (sr_file == NULL) {
return (DAT_INTERNAL_ERROR);
}
for (;;) {
if (DAT_SUCCESS == dat_sr_parse_eof(sr_file)) {
break;
} else if (DAT_SUCCESS == dat_sr_parse_entry(sr_file)) {
continue;
} else {
dat_os_assert(!"unable to parse static registry file");
break;
}
}
if (0 != dat_os_fclose(sr_file)) {
return (DAT_INTERNAL_ERROR);
}
return (DAT_SUCCESS);
}
/*
*
* Internal Function Definitions
*
*/
/*
* Function: dat_sr_is_valid_entry
*/
DAT_BOOLEAN
dat_sr_is_valid_entry(
DAT_SR_CONF_ENTRY *entry)
{
if ((DAT_SR_API_UDAT == entry->api_version.type) &&
(entry->is_default)) {
return (DAT_TRUE);
} else {
return (DAT_FALSE);
}
}
/*
* Function: dat_sr_load_entry
*/
DAT_RETURN
dat_sr_load_entry(
DAT_SR_CONF_ENTRY *conf_entry)
{
DAT_SR_ENTRY entry;
if (DAT_NAME_MAX_LENGTH < (strlen(conf_entry->ia_name) + 1)) {
dat_os_dbg_print(DAT_OS_DBG_TYPE_SR,
"DAT Registry: ia name %s is longer than "
"DAT_NAME_MAX_LENGTH (%i)\n",
conf_entry->ia_name, DAT_NAME_MAX_LENGTH);
return (DAT_INSUFFICIENT_RESOURCES);
}
(void) dat_os_strncpy(entry.info.ia_name, conf_entry->ia_name,
DAT_NAME_MAX_LENGTH);
entry.info.dapl_version_major = conf_entry->api_version.version.major;
entry.info.dapl_version_minor = conf_entry->api_version.version.minor;
entry.info.is_thread_safe = conf_entry->is_thread_safe;
entry.lib_path = conf_entry->lib_path;
entry.ia_params = conf_entry->ia_params;
entry.lib_handle = NULL;
entry.ref_count = 0;
dat_os_dbg_print(DAT_OS_DBG_TYPE_SR,
"DAT Registry: loading provider for %s\n",
conf_entry->ia_name);
return (dat_sr_insert(&entry.info, &entry));
}
/*
* Function: dat_sr_type_to_str
*/
char *
dat_sr_type_to_str(
DAT_SR_TOKEN_TYPE type)
{
static char *str_array[] = { "string", "eor", "eof" };
if ((type < 0) || (2 < type)) {
return ("error: invalid token type");
}
return (str_array[type]);
}
/*
* Function: dat_sr_parse_eof
*/
DAT_RETURN
dat_sr_parse_eof(
DAT_OS_FILE *file)
{
DAT_SR_TOKEN token;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_EOF == token.type) {
return (DAT_SUCCESS);
} else {
(void) dat_sr_put_token(file, &token);
return (DAT_INTERNAL_ERROR);
}
}
/*
* Function: dat_sr_parse_ia_name
*/
DAT_RETURN
dat_sr_parse_entry(
DAT_OS_FILE *file)
{
DAT_SR_CONF_ENTRY entry;
DAT_RETURN status;
(void) dat_os_memset(&entry, 0, sizeof (DAT_SR_CONF_ENTRY));
if ((DAT_SUCCESS == dat_sr_parse_ia_name(file, &entry)) &&
(DAT_SUCCESS == dat_sr_parse_api(file, &entry)) &&
(DAT_SUCCESS == dat_sr_parse_thread_safety(file, &entry)) &&
(DAT_SUCCESS == dat_sr_parse_default(file, &entry)) &&
(DAT_SUCCESS == dat_sr_parse_lib_path(file, &entry)) &&
(DAT_SUCCESS == dat_sr_parse_provider_version(file, &entry)) &&
(DAT_SUCCESS == dat_sr_parse_ia_params(file, &entry)) &&
(DAT_SUCCESS == dat_sr_parse_platform_params(file, &entry)) &&
(DAT_SUCCESS == dat_sr_parse_eoe(file, &entry))) {
dat_os_dbg_print(DAT_OS_DBG_TYPE_SR,
"\n"
"DAT Registry: entry \n"
" ia_name %s\n"
" api_version\n"
" type 0x%X\n"
" major.minor %d.%d\n"
" is_thread_safe %d\n"
" is_default %d\n"
" lib_path %s\n"
" provider_version\n"
" id %s\n"
" major.minor %d.%d\n"
" ia_params %s\n"
"\n",
entry.ia_name,
entry.api_version.type,
entry.api_version.version.major,
entry.api_version.version.minor,
entry.is_thread_safe,
entry.is_default,
entry.lib_path,
entry.provider_version.id,
entry.provider_version.version.major,
entry.provider_version.version.minor,
entry.ia_params);
if (DAT_TRUE == dat_sr_is_valid_entry(&entry)) {
/*
* The static registry configuration file may have
* multiple entries with the same IA name. The first
* entry will be installed in the static registry
* causing subsequent attempts to register the same IA
* name to fail. Therefore the return code from
* dat_sr_load_entry() is ignored.
*/
(void) dat_sr_load_entry(&entry);
}
status = DAT_SUCCESS;
} else { /* resync */
DAT_SR_TOKEN token;
/*
* The static registry format is specified in the DAT
* specification. While the registry file's contents may change
* between revisions of the specification, there is no way to
* determine the specification version to which the
* configuration file conforms. If an entry is found that does
* not match the expected format, the entry is discarded
* and the parsing of the file continues. There is no way to
* determine if the entry was an error or an entry confirming
* to an alternate version of specification.
*/
for (;;) {
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
status = DAT_INTERNAL_ERROR;
break;
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_SUCCESS;
break;
} else {
dat_os_free(token.value,
(sizeof (char) *
dat_os_strlen(token.value)) + 1);
continue;
}
}
}
/* free resources */
if (NULL != entry.ia_name) {
dat_os_free(entry.ia_name,
sizeof (char) * (dat_os_strlen(entry.ia_name) + 1));
}
if (NULL != entry.lib_path) {
dat_os_free(entry.lib_path,
sizeof (char) * (dat_os_strlen(entry.lib_path) + 1));
}
if (NULL != entry.provider_version.id) {
dat_os_free(entry.provider_version.id,
sizeof (char) *
(dat_os_strlen(entry.provider_version.id) + 1));
}
if (NULL != entry.ia_params) {
dat_os_free(entry.ia_params,
sizeof (char) * (dat_os_strlen(entry.ia_params) + 1));
}
return (status);
}
/*
* Function: dat_sr_parse_ia_name
*/
DAT_RETURN
dat_sr_parse_ia_name(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry)
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_INTERNAL_ERROR;
} else {
entry->ia_name = token.value;
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_parse_ia_name
*/
DAT_RETURN
dat_sr_parse_api(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry)
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_INTERNAL_ERROR;
} else if (DAT_SUCCESS != dat_sr_convert_api(
token.value, &entry->api_version)) {
status = DAT_INTERNAL_ERROR;
} else {
dat_os_free(token.value,
(sizeof (char) * dat_os_strlen(token.value)) + 1);
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_parse_thread_safety
*/
static DAT_RETURN
dat_sr_parse_thread_safety(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry)
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_INTERNAL_ERROR;
} else if (DAT_SUCCESS != dat_sr_convert_thread_safety(
token.value, &entry->is_thread_safe)) {
status = DAT_INTERNAL_ERROR;
} else {
dat_os_free(token.value,
(sizeof (char) * dat_os_strlen(token.value)) + 1);
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_parse_default
*/
DAT_RETURN
dat_sr_parse_default(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry)
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_INTERNAL_ERROR;
} else if (DAT_SUCCESS != dat_sr_convert_default(
token.value, &entry->is_default)) {
status = DAT_INTERNAL_ERROR;
} else {
dat_os_free(token.value,
(sizeof (char) * dat_os_strlen(token.value)) + 1);
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_parse_lib_path
*/
DAT_RETURN
dat_sr_parse_lib_path(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry)
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_INTERNAL_ERROR;
} else {
entry->lib_path = token.value;
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_parse_provider_version
*/
DAT_RETURN
dat_sr_parse_provider_version(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry)
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_INTERNAL_ERROR;
} else if (DAT_SUCCESS != dat_sr_convert_provider_version(
token.value, &entry->provider_version)) {
status = DAT_INTERNAL_ERROR;
} else {
dat_os_free(token.value,
(sizeof (char) * dat_os_strlen(token.value)) + 1);
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_parse_ia_params
*/
DAT_RETURN
dat_sr_parse_ia_params(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry)
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_INTERNAL_ERROR;
} else {
entry->ia_params = token.value;
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_parse_platform_params
*/
DAT_RETURN
dat_sr_parse_platform_params(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry)
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if (DAT_SR_TOKEN_STRING != token.type) {
status = DAT_INTERNAL_ERROR;
} else {
entry->platform_params = token.value;
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_parse_eoe
*/
DAT_RETURN
dat_sr_parse_eoe(
DAT_OS_FILE *file,
DAT_SR_CONF_ENTRY *entry) /*ARGSUSED*/
{
DAT_SR_TOKEN token;
DAT_RETURN status;
if (DAT_SUCCESS != dat_sr_get_token(file, &token)) {
return (DAT_INTERNAL_ERROR);
}
if ((DAT_SR_TOKEN_EOF != token.type) &&
(DAT_SR_TOKEN_EOR != token.type)) {
status = DAT_INTERNAL_ERROR;
} else {
status = DAT_SUCCESS;
}
if (DAT_SUCCESS != status) {
DAT_RETURN status_success;
status_success = dat_sr_put_token(file, &token);
dat_os_assert(DAT_SUCCESS == status_success);
}
return (status);
}
/*
* Function: dat_sr_convert_api
*/
DAT_RETURN
dat_sr_convert_api(
char *str,
DAT_SR_API_VERSION *api_version)
{
int i;
int minor_i;
dat_os_assert(0 < dat_os_strlen(str));
if ('u' == str[0]) {
api_version->type = DAT_SR_API_UDAT;
} else if ('k' == str[0]) {
api_version->type = DAT_SR_API_KDAT;
} else {
return (DAT_INTERNAL_ERROR);
}
for (i = 1 /* move past initial [u|k] */; '\0' != str[i]; i++) {
if ('.' == str[i]) {
break;
} else if (DAT_TRUE != dat_os_isdigit(str[i])) {
return (DAT_INTERNAL_ERROR);
}
}
api_version->version.major = (DAT_UINT32)dat_os_strtol(str + 1, NULL,
10);
/* move past '.' */
minor_i = ++i;
for (; '\0' != str[i]; i++) {
if (DAT_TRUE != dat_os_isdigit(str[i])) {
return (DAT_INTERNAL_ERROR);
}
}
api_version->version.minor = (DAT_UINT32)dat_os_strtol(str + minor_i,
NULL, 10);
if ('\0' != str[i]) {
return (DAT_INTERNAL_ERROR);
}
return (DAT_SUCCESS);
}
/*
* Function: dat_sr_convert_thread_safety
*/
static DAT_RETURN
dat_sr_convert_thread_safety(
char *str,
DAT_BOOLEAN *is_thread_safe)
{
if (!dat_os_strncmp(str,
DAT_SR_TOKEN_THREADSAFE,
dat_os_strlen(DAT_SR_TOKEN_THREADSAFE))) {
*is_thread_safe = DAT_TRUE;
return (DAT_SUCCESS);
} else if (!dat_os_strncmp(str,
DAT_SR_TOKEN_NONTHREADSAFE,
dat_os_strlen(DAT_SR_TOKEN_NONTHREADSAFE))) {
*is_thread_safe = DAT_FALSE;
return (DAT_SUCCESS);
} else {
return (DAT_INTERNAL_ERROR);
}
}
/*
* Function: dat_sr_convert_default
*/
static DAT_RETURN
dat_sr_convert_default(
char *str,
DAT_BOOLEAN *is_default)
{
if (!dat_os_strncmp(str,
DAT_SR_TOKEN_DEFAULT,
dat_os_strlen(DAT_SR_TOKEN_DEFAULT))) {
*is_default = DAT_TRUE;
return (DAT_SUCCESS);
} else if (!dat_os_strncmp(str,
DAT_SR_TOKEN_NONDEFAULT,
dat_os_strlen(DAT_SR_TOKEN_NONDEFAULT))) {
*is_default = DAT_FALSE;
return (DAT_SUCCESS);
} else {
return (DAT_INTERNAL_ERROR);
}
}
/*
* Function: dat_sr_convert_provider_version
*/
DAT_RETURN
dat_sr_convert_provider_version(
char *str,
DAT_SR_PROVIDER_VERSION *provider_version)
{
DAT_RETURN status;
int i;
int decimal_i;
dat_os_assert(0 < dat_os_strlen(str));
dat_os_assert(NULL == provider_version->id);
status = DAT_SUCCESS;
for (i = 0; '\0' != str[i]; i++) {
if ('.' == str[i]) {
break;
}
}
/* if no id value was found */
if (0 == i) {
status = DAT_INTERNAL_ERROR;
goto exit;
}
if (NULL == (provider_version->id = dat_os_alloc(sizeof (char) *
(i + 1)))) {
status = DAT_INSUFFICIENT_RESOURCES | DAT_RESOURCE_MEMORY;
goto exit;
}
(void) dat_os_strncpy(provider_version->id, str, i);
provider_version->id[i] = '\0';
/* move past '.' */
decimal_i = ++i;
for (; '\0' != str[i]; i++) {
if ('.' == str[i]) {
break;
} else if (DAT_TRUE != dat_os_isdigit(str[i])) {
status = DAT_INTERNAL_ERROR;
goto exit;
}
}
/* if no version value was found */
if (decimal_i == i) {
status = DAT_INTERNAL_ERROR;
goto exit;
}
provider_version->version.major = (DAT_UINT32)
dat_os_strtol(str + decimal_i, NULL, 10);
/* move past '.' */
decimal_i = ++i;
for (; '\0' != str[i]; i++) {
if (DAT_TRUE != dat_os_isdigit(str[i])) {
status = DAT_INTERNAL_ERROR;
goto exit;
}
}
/* if no version value was found */
if (decimal_i == i) {
status = DAT_INTERNAL_ERROR;
goto exit;
}
provider_version->version.minor = (DAT_UINT32)
dat_os_strtol(str + decimal_i, NULL, 10);
if ('\0' != str[i]) {
status = DAT_INTERNAL_ERROR;
goto exit;
}
exit:
if (DAT_SUCCESS != status) {
if (NULL != provider_version->id) {
dat_os_free(provider_version->id,
sizeof (char) *
(dat_os_strlen(provider_version->id) + 1));
provider_version->id = NULL;
}
}
return (status);
}
/*
* Function: dat_sr_get_token
*/
DAT_RETURN
dat_sr_get_token(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token)
{
if (NULL == g_token_stack) {
return (dat_sr_read_token(file, token));
} else {
DAT_SR_STACK_NODE *top;
top = g_token_stack;
*token = top->token;
g_token_stack = top->next;
dat_os_free(top, sizeof (DAT_SR_STACK_NODE));
return (DAT_SUCCESS);
}
}
/*
* Function: dat_sr_put_token
*/
DAT_RETURN
dat_sr_put_token(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token) /*ARGSUSED*/
{
DAT_SR_STACK_NODE *top;
if (NULL == (top = dat_os_alloc(sizeof (DAT_SR_STACK_NODE)))) {
return (DAT_INSUFFICIENT_RESOURCES | DAT_RESOURCE_MEMORY);
}
top->token = *token;
top->next = g_token_stack;
g_token_stack = top;
return (DAT_SUCCESS);
}
/*
* Function: dat_sr_read_token
*/
DAT_RETURN
dat_sr_read_token(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token)
{
DAT_OS_FILE_POS pos;
DAT_OS_SIZE token_len;
DAT_COUNT num_escape_seq;
DAT_BOOLEAN is_quoted_str;
DAT_BOOLEAN is_prev_char_backslash;
/*
* The DAT standard does not specify a maximum size for quoted strings.
* Therefore the tokenizer must be able to read in a token of arbitrary
* size. Instead of allocating a fixed length buffer, the tokenizer
* first scans the input a single character at a time looking for the
* begining and end of the token. Once the these positions are found,
* the entire token is read into memory. By using this algorithm, the
* implementation does not place an arbitrary maximum on the token size.
*/
token_len = 0;
num_escape_seq = 0;
is_quoted_str = DAT_FALSE;
is_prev_char_backslash = DAT_FALSE;
for (;;) {
DAT_OS_FILE_POS cur_pos;
int c;
/* if looking for start of the token */
if (0 == token_len) {
if (DAT_SUCCESS != dat_os_fgetpos(file, &cur_pos)) {
return (DAT_INTERNAL_ERROR);
}
}
c = dat_os_fgetc(file);
/* if looking for start of the token */
if (0 == token_len) {
if (EOF == c) {
token->type = DAT_SR_TOKEN_EOF;
token->value = NULL;
token->value_len = 0;
goto success;
} else if (DAT_SR_CHAR_NEWLINE == c) {
token->type = DAT_SR_TOKEN_EOR;
token->value = NULL;
token->value_len = 0;
goto success;
} else if (dat_os_isblank(c)) {
continue;
} else if (DAT_SR_CHAR_COMMENT == c) {
dat_sr_read_comment(file);
continue;
} else {
if (DAT_SR_CHAR_QUOTE == c) {
is_quoted_str = DAT_TRUE;
}
pos = cur_pos;
token_len++;
}
} else { /* looking for the end of the token */
if (EOF == c) {
break;
} else if (DAT_SR_CHAR_NEWLINE == c) {
/* put back the newline */
(void) dat_os_fungetc(file);
break;
} else if (!is_quoted_str && dat_os_isblank(c)) {
break;
} else {
token_len++;
if ((DAT_SR_CHAR_QUOTE == c) &&
!is_prev_char_backslash) {
break;
} else if ((DAT_SR_CHAR_BACKSLASH == c) &&
!is_prev_char_backslash) {
is_prev_char_backslash = DAT_TRUE;
num_escape_seq++;
} else {
is_prev_char_backslash = DAT_FALSE;
}
}
}
}
/* the token was a string */
if (DAT_SUCCESS != dat_os_fsetpos(file, &pos)) {
return (DAT_INTERNAL_ERROR);
}
if (is_quoted_str) {
if (DAT_SUCCESS != dat_sr_read_quoted_str(file,
token,
token_len,
num_escape_seq)) {
return (DAT_INTERNAL_ERROR);
}
} else {
if (DAT_SUCCESS != dat_sr_read_str(file,
token,
token_len)) {
return (DAT_INTERNAL_ERROR);
}
}
success:
dat_os_dbg_print(DAT_OS_DBG_TYPE_SR,
"\n"
"DAT Registry: token\n"
" type %s\n"
" value <%s>\n"
"\n",
dat_sr_type_to_str(token->type),
((DAT_SR_TOKEN_STRING == token->type) ? token->value : ""));
return (DAT_SUCCESS);
}
/*
* Function: dat_sr_read_str
*/
DAT_RETURN
dat_sr_read_str(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token,
DAT_OS_SIZE token_len)
{
token->type = DAT_SR_TOKEN_STRING;
/* +1 for null termination */
token->value_len = sizeof (char) * (token_len + 1);
if (NULL == (token->value = dat_os_alloc(token->value_len))) {
return (DAT_INSUFFICIENT_RESOURCES | DAT_RESOURCE_MEMORY);
}
if (token_len != dat_os_fread(file, token->value, token_len)) {
dat_os_free(token->value, token->value_len);
token->value = NULL;
return (DAT_INTERNAL_ERROR);
}
token->value[token->value_len - 1] = '\0';
return (DAT_SUCCESS);
}
/*
* Function: dat_sr_read_quoted_str
*/
DAT_RETURN
dat_sr_read_quoted_str(
DAT_OS_FILE *file,
DAT_SR_TOKEN *token,
DAT_OS_SIZE token_len,
DAT_COUNT num_escape_seq)
{
DAT_OS_SIZE str_len;
DAT_OS_SIZE i;
DAT_OS_SIZE j;
int c;
DAT_RETURN status;
DAT_BOOLEAN is_prev_char_backslash;
str_len = token_len - 2; /* minus 2 " characters */
is_prev_char_backslash = DAT_FALSE;
status = DAT_SUCCESS;
token->type = DAT_SR_TOKEN_STRING;
/* +1 for null termination */
token->value_len = sizeof (char) * (str_len - num_escape_seq + 1);
if (NULL == (token->value = dat_os_alloc(token->value_len))) {
status = DAT_INSUFFICIENT_RESOURCES | DAT_RESOURCE_MEMORY;
goto exit;
}
/* throw away " */
if (DAT_SR_CHAR_QUOTE != dat_os_fgetc(file)) {
status = DAT_INTERNAL_ERROR;
goto exit;
}
for (i = 0, j = 0; i < str_len; i++) {
c = dat_os_fgetc(file);
if (EOF == c) {
status = DAT_INTERNAL_ERROR;
goto exit;
} else if ((DAT_SR_CHAR_BACKSLASH == c) &&
!is_prev_char_backslash) {
is_prev_char_backslash = DAT_TRUE;
} else {
token->value[j] = c;
j++;
is_prev_char_backslash = DAT_FALSE;
}
}
/* throw away " */
if (DAT_SR_CHAR_QUOTE != dat_os_fgetc(file)) {
status = DAT_INTERNAL_ERROR;
goto exit;
}
token->value[token->value_len - 1] = '\0';
exit:
if (DAT_SUCCESS != status) {
if (NULL != token->value) {
dat_os_free(token->value, token->value_len);
token->value = NULL;
}
}
return (status);
}
/*
* Function: dat_sr_read_comment
*/
void
dat_sr_read_comment(
DAT_OS_FILE *file)
{
int c;
/* read up to an EOR or EOF to move past the comment */
do {
c = dat_os_fgetc(file);
} while ((DAT_SR_CHAR_NEWLINE != c) && (EOF != c));
/* put back the newline */
(void) dat_os_fungetc(file);
}