You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ardix/kernel/serial.c

114 lines
2.4 KiB
C

/* See the end of this file for copyright, license, and warranty information. */
#include <ardix/atomic.h>
#include <ardix/device.h>
#include <ardix/mutex.h>
#include <ardix/ringbuf.h>
#include <ardix/sched.h>
#include <ardix/serial.h>
#include <arch/serial.h>
#include <config.h>
#include <stddef.h>
static ssize_t serial_device_read(void *dest, struct device *dev, size_t len, off_t offset)
{
ssize_t ret;
struct serial_device *serial_dev = container_of(dev, struct serial_device, device);
ret = mutex_trylock(&dev->lock);
if (ret == 0) {
ret = serial_read(dest, serial_dev, len);
mutex_unlock(&dev->lock);
}
return ret;
}
static ssize_t serial_device_write(struct device *dev, const void *src, size_t len, off_t offset)
{
ssize_t ret;
struct serial_device *serial_dev = container_of(dev, struct serial_device, device);
ret = mutex_trylock(&dev->lock);
if (ret == 0) {
ret = serial_write(serial_dev, src, len);
mutex_unlock(&dev->lock);
}
return ret;
}
int serial_init(struct serial_device *dev, long int baud)
{
int err = -1;
if (dev->id < 0)
return -1; /* invalid dev */
dev->device.read = serial_device_read;
dev->device.write = serial_device_write;
err = device_init(&dev->device);
if (err)
goto err_device_init;
dev->baud = baud;
dev->rx = ringbuf_create(CONFIG_SERIAL_BUFSZ);
if (dev->rx == NULL)
goto err_ringbuf_create;
err = arch_serial_init(dev);
if (err)
goto err_arch_serial_init;
goto out;
err_arch_serial_init:
ringbuf_destroy(dev->rx);
err_ringbuf_create:
device_put(&dev->device);
err_device_init:
out:
return err;
}
void serial_exit(struct serial_device *dev)
{
arch_serial_exit(dev);
ringbuf_destroy(dev->rx);
dev->id = -1;
}
ssize_t serial_read(void *dest, struct serial_device *dev, size_t len)
{
ssize_t ret;
ret = (ssize_t)ringbuf_read(dest, dev->rx, len);
return ret;
}
ssize_t serial_write(struct serial_device *dev, const void *data, size_t len)
{
ssize_t ret;
ret = arch_serial_write(dev, data, len);
return ret;
}
/*
* This file is part of Ardix.
* Copyright (c) 2020, 2021 Felix Kopp <owo@fef.moe>.
*
* Ardix is non-violent software: you may only use, redistribute,
* and/or modify it under the terms of the CNPLv6+ as found in
* the LICENSE file in the source code root directory or at
* <https://git.pixie.town/thufie/CNPL>.
*
* Ardix comes with ABSOLUTELY NO WARRANTY, to the extent
* permitted by applicable law. See the CNPLv6+ for details.
*/