/* 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.
 */