a0e6850f82
No Minix-specific changes needed. Change-Id: Ia8ddbdb57ac04dfb42d79c374b9e25b189f9dc3b
274 lines
6.3 KiB
C
274 lines
6.3 KiB
C
/* $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;
|
|
}
|