NetBSD-5.0.2/lib/libform/field_types.c

Compare this file to the similar file:
Show the results in this format:

/*	$NetBSD: field_types.c,v 1.7 2006/03/19 20:02:27 christos Exp $	*/

/*-
 * Copyright (c) 1998-1999 Brett Lymn
 *                         (blymn@baea.com.au, brett_lymn@yahoo.com.au)
 * All rights reserved.
 *
 * This code has been donated to The NetBSD Foundation by the Author.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. The name of the author may not be used to endorse or promote products
 *    derived from this software without specific prior written permission
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 *
 *
 */

#include <sys/cdefs.h>
__RCSID("$NetBSD: field_types.c,v 1.7 2006/03/19 20:02:27 christos Exp $");

#include <stdlib.h>
#include <stdarg.h>
#include "form.h"
#include "internals.h"

extern FIELD _formi_default_field;

/* function prototypes.... */
static void
_formi_create_field_args(FIELDTYPE *type, char **type_args,
			 formi_type_link **link, va_list *args, int *error);
static FIELDTYPE *
_formi_create_fieldtype(void);

/*
 * Process the arguments, if any, for the field type.
 */
static void
_formi_create_field_args(FIELDTYPE *type, char **type_args,
			 formi_type_link **link, va_list *args, int *error)
{
	formi_type_link *l;

	l = NULL;
	if ((type != NULL)
	    && ((type->flags & _TYPE_HAS_ARGS) == _TYPE_HAS_ARGS)) {
		if ((type->flags & _TYPE_IS_LINKED) == _TYPE_IS_LINKED) {
			l = malloc(sizeof(*l));
			if (l != NULL) {
				_formi_create_field_args(type->link->next,
							 type_args,
							 &type->link->next->link,
							 args,
							 error);
				_formi_create_field_args(type->link->prev,
							 type_args,
							 &type->link->prev->link,
							 args,
							 error);
				(*link) = l;
			}

			(*error)++;
		} else {
			if ((*type_args = (char *) type->make_args(args))
			    == NULL)
				(*error)++;
		}
	}
}

/*
 * Allocate a new fieldtype structure, initialise it and return the
 * struct to the caller.
 */
static FIELDTYPE *
_formi_create_fieldtype(void)
{
	FIELDTYPE *new;
	
	if ((new = malloc(sizeof(*new))) == NULL)
		return NULL;

	new->flags = _TYPE_NO_FLAGS;
	new->refcount = 0;
	new->link = NULL;
	new->make_args = NULL;
	new->copy_args = NULL;
	new->free_args = NULL;
	new->field_check = NULL;
	new->char_check = NULL;
	new->next_choice = NULL;
	new->prev_choice = NULL;

	return new;
}

/*
 * Set the field type of the field to be the one given.
 */
int
set_field_type(FIELD *fptr, FIELDTYPE *type, ...)
{
	va_list args;
	FIELD *field;
	int error = 0;

	va_start(args, type);
	
	field = (fptr == NULL)? &_formi_default_field : fptr;

	field->type = type;
	_formi_create_field_args(type, &field->args, &type->link, &args,
				 &error);
	va_end(args);
	
	if (error)
		return E_BAD_ARGUMENT;
		
	return E_OK;
}

/*
 * Return the field type associated with the given field
 */
FIELDTYPE *
field_type(FIELD *fptr)
{
	FIELD *field;
	
	field = (fptr == NULL)? &_formi_default_field : fptr;

	return field->type;
}


/*
 * Return the field arguments for the given field.
 */
char *
field_arg(FIELD *fptr)
{
	FIELD *field;

	field = (fptr == NULL)? &_formi_default_field : fptr;

	return field->args;
}

/*
 * Create a new field type.  Caller must specify a field_check routine
 * and char_check routine.
 */
FIELDTYPE *
new_fieldtype(int (*field_check)(FIELD *, char *),
	      int (*char_check)(int, char *))
{
	FIELDTYPE *new;
	
	if ((field_check == NULL) && (char_check == NULL))
		return NULL;

	if ((new = _formi_create_fieldtype()) != NULL) {
		new->field_check = field_check;
		new->char_check = char_check;
	}

	return new;
}

/*
 * Free the storage used by the fieldtype.
 */
int
free_fieldtype(FIELDTYPE *fieldtype)
{
	if (fieldtype == NULL)
		return E_BAD_ARGUMENT;

	if (fieldtype->refcount > 0)
		return E_CONNECTED;

	if ((fieldtype->flags & _TYPE_IS_BUILTIN) == _TYPE_IS_BUILTIN)
		return E_BAD_ARGUMENT; /* don't delete builtin types! */

	if ((fieldtype->flags & _TYPE_IS_LINKED) == _TYPE_IS_LINKED) 
	{
		fieldtype->link->next->refcount--;
		fieldtype->link->prev->refcount--;
	}
	
	free(fieldtype);

	return E_OK;
}

/*
 * Set the field type arguments for the given field type.
 */
int
set_fieldtype_arg(FIELDTYPE *fieldtype, char * (*make_args)(va_list *),
		  char * (*copy_args)(char*), void (*free_args)(char *))
{
	if ((fieldtype == NULL) || (make_args == NULL)
	    || (copy_args == NULL) || (free_args == NULL))
		return E_BAD_ARGUMENT;

	fieldtype->make_args = make_args;
	fieldtype->copy_args = copy_args;
	fieldtype->free_args = free_args;

	return E_OK;
}

/*
 * Set up the choice list functions for the given fieldtype.
 */
int
set_fieldtype_choice(FIELDTYPE *fieldtype, int (*next_choice)(FIELD *, char *),
		     int (*prev_choice)(FIELD *, char *))
{
	if ((fieldtype == NULL) || (next_choice == NULL)
	    || (prev_choice == NULL))
		return E_BAD_ARGUMENT;

	fieldtype->next_choice = next_choice;
	fieldtype->prev_choice = prev_choice;

	return E_OK;
}

/*
 * Link the two given types to produce a new type, return this new type.
 */
FIELDTYPE *
link_fieldtype(FIELDTYPE *type1, FIELDTYPE *type2)
{
	FIELDTYPE *new;

	if ((type1 == NULL) || (type2 == NULL))
		return NULL;
	
	if ((new = _formi_create_fieldtype()) == NULL)
		return NULL;

	new->flags = _TYPE_IS_LINKED;
	new->flags |= ((type1->flags & _TYPE_HAS_ARGS)
		       | (type2->flags & _TYPE_HAS_ARGS));
	if ((new->link = malloc(sizeof(*new->link))) == NULL) {
		free(new);
		return NULL;
	}
	
	new->link->prev = type1;
	new->link->next = type2;
	type1->refcount++;
	type2->refcount++;
	
	return new;
}