OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [branches/] [stable_0_2_x/] [or1ksim/] [peripheral/] [channels/] [channel.c] - Rev 1549

Go to most recent revision | Compare with Previous | Blame | View Log

/* channel.c -- Definition of types and structures for 
   peripherals to communicate with host.  Adapted from UML.
 
   Copyright (C) 2002 Richard Prescott <rip@step.polymtl.ca>
 
This file is part of OpenRISC 1000 Architectural Simulator.
 
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
 
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.
 
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
 
#if HAVE_CONFIG_H
#include "config.h"
#endif
 
#define _GNU_SOURCE	/* for strndup */
 
#include <stdio.h>	/* perror */
#include <stdlib.h>	/* exit */
 
#if HAVE_MALLOC_H
#include <malloc.h>	/* calloc, free */
#endif
 
#include <string.h>	/* strndup, strcmp, strlen, strchr */
#include <errno.h>	/* errno */
 
#include "port.h"
 
#include "channel.h"
 
struct channel_factory
{
	const char * name;
	const struct channel_ops * ops;
	struct channel_factory * next;
};
 
extern struct channel_ops fd_channel_ops, file_channel_ops,
	xterm_channel_ops, tcp_channel_ops, tty_channel_ops;
 
static struct channel_factory preloaded[] =
{
	{ "fd",     &fd_channel_ops,     &preloaded[1] },
	{ "file",   &file_channel_ops,   &preloaded[2] },
	{ "xterm",  &xterm_channel_ops,  &preloaded[3] },
	{ "tcp",    &tcp_channel_ops,    &preloaded[4] },
	{ "tty",    &tty_channel_ops,    NULL          }
};
 
static struct channel_factory * head = &preloaded[0];
 
static struct channel_factory * find_channel_factory(const char * name);
 
struct channel * channel_init(const char * descriptor)
{
	struct channel * retval;
	struct channel_factory * current;
	char * args, * name;
	int count;
 
	if(!descriptor)
	{
		return NULL;
	}
 
	retval = (struct channel*)calloc(1, sizeof(struct channel));
 
	if(!retval)
	{
		perror(descriptor);
		exit(1);
	}
 
	args = strchr(descriptor, ':');
 
	if(args)
	{
		count = args - descriptor;
		args++;
	}
	else
	{
		count = strlen(descriptor);
	}
 
	name = (char*)strndup(descriptor, count);
 
	if(!name)
	{
		perror(name);
		exit(1);
	}
 
	current = find_channel_factory(name);
 
	if(!current)
	{
		errno = ENODEV;
		perror(descriptor);
		exit(1);
	}
 
	retval->ops = current->ops;
 
	free(name);
 
	if(!retval->ops)
	{
		errno = ENODEV;
		perror(descriptor);
		exit(1);
	}
 
	if(retval->ops->init)
	{
		retval->data = (retval->ops->init)(args);
 
		if(!retval->data)
		{
			perror(descriptor);
			exit(1);
		}
	}
 
	return retval;
}
 
int channel_open(struct channel * channel)
{
	if(channel && channel->ops && channel->ops->open)
	{
		return (channel->ops->open)(channel->data);
	}
	errno = ENOSYS;
	return -1;
}
 
int channel_read(struct channel * channel, char * buffer, int size)
{
	if(channel && channel->ops && channel->ops->read)
	{
		return (channel->ops->read)(channel->data, buffer, size);
	}
	errno = ENOSYS;
	return -1;
}
 
int channel_write(struct channel * channel, const char * buffer, int size)
{
	if(channel && channel->ops && channel->ops->write)
	{
		return (channel->ops->write)(channel->data, buffer, size);
	}
	errno = ENOSYS;
	return -1;
}
 
void channel_close(struct channel * channel)
{
	if(channel && channel->ops && channel->ops->close)
	{
		(channel->ops->close)(channel->data);
	}
}
 
void channel_free(struct channel * channel)
{
	if(channel && channel->ops && channel->ops->free)
	{
		(channel->ops->free)(channel->data);
		free(channel);
	}
}
 
 
int channel_ok(struct channel * channel)
{
	if(channel && channel->ops)
	{
		if(channel->ops->isok)
			return (channel->ops->isok)(channel->data);
		else
			return 1;
	}
	return 0;
}
 
char * channel_status(struct channel * channel)
{
	if(channel && channel->ops && channel->ops->status)
	{
		return (channel->ops->status)(channel->data);
	}
	return "";
}
 
 
 
static struct channel_factory * find_channel_factory(const char * name)
{
	struct channel_factory * current = head;
 
	current = head;
	while(current && strcmp(current->name, name))
	{
		current = current->next;
	}
 
	return current;
}
 
int register_channel(const char * name, const struct channel_ops * ops)
{
	struct channel_factory * new;
 
 
	if(find_channel_factory(name))
	{
		errno = EEXIST;
		perror(name);
		exit(1);
	}
 
	new = (struct channel_factory *)calloc(1, sizeof(struct channel_factory));
 
	if(!new)
	{
		perror(name);
		exit(1);
	}
 
	new->name = name;
	new->ops  = ops;
	new->next = head;
	head = new;
 
	return (int)new;	/* dummy */
}
 
/*
 * Local variables:
 * c-file-style: "linux"
 * End:
 */
 
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.