/*-
* Copyright (c) 2002-2018 The UbixOS Project.
* All rights reserved.
*
* This was developed by Christopher W. Olsen for the UbixOS Project.
*
* 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, the following disclaimer and the list of authors.
* 2) Redistributions in binary form must reproduce the above copyright notice, this list of
* conditions, the following disclaimer and the list of authors in the documentation and/or
* other materials provided with the distribution.
* 3) Neither the name of the UbixOS Project nor the names of its contributors may be used to
* endorse or promote products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS "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 COPYRIGHT OWNER OR CONTRIBUTORS
* 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/descrip.h>
#include <sys/sysproto_posix.h>
#include <sys/thread.h>
#include <lib/kprintf.h>
#include <ubixos/endtask.h>
#include <lib/kmalloc.h>
#include <assert.h>
#include <sys/select.h>
// XXX MrOlsen (2020-01-30) No longer needed -> static struct file *kern_files = 0x0;
int fcntl(struct thread *td, struct sys_fcntl_args *uap) {
struct file *fp = 0x0;
struct file *dup_fp = 0x0;
int i = 0;
if (td->o_files[uap->fd] == 0x0) {
kprintf("ERROR!!!\n");
return (-1);
}
kprintf("CMD: %i", uap->cmd);
fp = (struct file*) td->o_files[uap->fd];
switch (uap->cmd) {
case 17:
/* First 5 Descriptors Are Reserved */
for (i = 5; i < MAX_FILES; i++) {
if (td->o_files[i] == 0x0) {
dup_fp = (struct file*) kmalloc(sizeof(struct file));
memcpy(dup_fp, fp, sizeof(struct file));
td->o_files[i] = (void*) dup_fp;
td->td_retval[0] = i;
((struct file*) td->o_files[uap->fd])->fd->dup++;
fclose(((struct file*) td->o_files[uap->fd])->fd);
//td->o_files[uap->fd] = 0;
if (fdestroy(td, fp, uap->fd) != 0x0)
kprintf("[%s:%i] fdestroy(0x%X, 0x%X) failed\n", __FILE__, __LINE__, fp, td->o_files[uap->fd]);
kprintf("FCNTL: %i, %i, 0x%X.", i, uap->fd, fp);
break;
}
}
break;
case 3:
if (uap->fd > 3)
fp->f_flag = O_RDWR & O_ACCMODE;
td->td_retval[0] = fp->f_flag;
break;
case 4:
fp->f_flag &= ~FCNTLFLAGS;
fp->f_flag |= FFLAGS(uap->arg & ~O_ACCMODE) & FCNTLFLAGS;
break;
default:
kprintf("ERROR DEFAULT: [%i]", uap->fd);
}
return (0x0);
}
int sys_fcntl(struct thread *td, struct sys_fcntl_args *uap) {
return (fcntl(td, uap));
}
int falloc(struct thread *td, struct file **resultfp, int *resultfd) {
struct file *fp = 0x0;
int i = 0;
fp = (struct file*) kmalloc(sizeof(struct file));
/* First 5 Descriptors Are Reserved */
for (i = 5; i < MAX_FILES; i++) {
if (td->o_files[i] == 0x0) {
td->o_files[i] = (void*) fp;
if (resultfd)
*resultfd = i;
if (resultfp)
*resultfp = fp;
goto allocated;
}
}
kfree(fp);
*resultfp = 0x0;
*resultfd = 0x0;
allocated:
return (0x0);
}
#include <sys/ioctl.h>
/**
* @brief This destroys a thread local file descriptor.
*
* @details This function will destroy the thread local file file specified as fd,
* for sanity it requires the file descriptor id and memory address to check before destroying.
*
* @note This text shall only show you, how such a \"note\" section
* is looking. There is nothing which really needs your notice,
* so you do not really need to read this section.
*
* @param td Pointer to thread which initiated the syscall.
* @param fp Pointer to the thread local file which you want to destroy.
* @param fd File descriptor id that points to the thread local file which you want to destroy.
*
* @return The error return code of the function.
*
* @retval 0 The function is successfully executed
* @retval -1 An error occurred
*/
int fdestroy(struct thread *td, struct file *fp, int fd) {
int error = 0;
if (td->o_files[fd] != fp) {
error = -1;
}
else {
kfree(td->o_files[fd]);
td->o_files[fd] = 0x0;
}
return (error);
}
int close(struct thread *td, struct close_args *uap) {
#ifdef DEBUG
kprintf("[%s:%i]",__FILE__,__LINE__);
#endif
kprintf("[%s:%i]", __FILE__, __LINE__);
kfree((void*) td->o_files[uap->fd]);
td->o_files[uap->fd] = 0x0;
td->td_retval[0] = 0x0;
return (0x0);
}
/*!
* \brief return data table size
*/
int getdtablesize(struct thread *td, struct getdtablesize_args *uap) {
#ifdef DEBUG
kprintf("[%s:%i]",__FILE__,__LINE__);
#endif
td->td_retval[0] = O_FILES;
return (0);
}
/* HACK */
int fstat(struct thread *td, struct sys_fstat_args *uap) {
struct file *fp = 0x0;
#ifdef DEBUG
kprintf("[%s:%i]",__FILE__,__LINE__);
#endif
fp = (struct file*) _current->td.o_files[uap->fd];
uap->sb->st_mode = 0x2180;
uap->sb->st_blksize = 0x1000;
kprintf("fstat: %i", uap->fd);
return (0x0);
}
/*!
* \brief ioctl functionality not implimented yet
*
* \returns NULL for now
*/
int ioctl(struct thread *td, struct ioctl_args *uap) {
td->td_retval[0] = 0x0;
return (0x0);
}
/*!
* \brief get pointer to file fd in specified thread
*
* \return returns fp
*/
int getfd(struct thread *td, struct file **fp, int fd) {
int error = 0x0;
#ifdef DEBUG
kprintf("[%s:%i]",__FILE__,__LINE__);
#endif
*fp = (struct file*) td->o_files[fd];
if (fp == 0x0)
error = -1;
return (error);
}
int sys_ioctl(struct thread *td, struct sys_ioctl_args *args) {
switch (args->com) {
case TIOCGETA:
if (args->fd == 0 || args->fd == 1) {
struct termios *t = (struct termios*) args->data;
t->c_iflag = 0x2B02;
t->c_oflag = 0x3;
t->c_cflag = 0x4B00;
t->c_lflag = 0x5CB;
t->c_cc[0] = 4;
t->c_cc[1] = 255;
t->c_cc[2] = 255;
t->c_cc[3] = 127;
t->c_cc[4] = 23;
t->c_cc[5] = 21;
t->c_cc[6] = 18;
t->c_cc[7] = 8;
t->c_cc[8] = 3;
t->c_cc[9] = 28;
t->c_cc[10] = 26;
t->c_cc[11] = 25;
t->c_cc[12] = 17;
t->c_cc[13] = 19;
t->c_cc[14] = 22;
t->c_cc[15] = 15;
t->c_cc[16] = 1;
t->c_cc[17] = 0;
t->c_cc[18] = 20;
t->c_cc[19] = 255;
t->c_ispeed = 0x9600;
t->c_ospeed = 0x9600;
td->td_retval[0] = 0;
}
else {
td->td_retval[0] = -1;
}
break;
case TIOCGWINSZ:
asm("nop");
struct winsize *win = (struct winsize*) args->data;
win->ws_row = 50;
win->ws_col = 80;
break;
default:
kprintf("ioFD:%i:0x%X!", args->fd, args->com);
break;
}
td->td_retval[0] = 0x0;
return (0x0);
}
int sys_select(struct thread *td, struct sys_select_args *args) {
int error = 0x0;
/*
int i = 0x0;
fd_set sock_rfds;
fd_set sock_wfds;
fd_set sock_efds;
FD_ZERO(&sock_rfds);
FD_ZERO(&sock_wfds);
FD_ZERO(&sock_efds);
if (args->in != 0x0) {
for (i = 0; i < args->nd; i++) {
FD_SET(((struct file * ) td->o_files[args->in[0]]).socket, &sock_rfds);
}
}
if (args->ou != 0x0) {
for (i = 0; i < args->nd; i++) {
FD_SET(((struct file * ) td->o_files[args->ou[0]]).socket, &sock_wfds);
}
}
if (args->ex != 0x0) {
for (i = 0; i < args->nd; i++) {
FD_SET(((struct file * ) td->o_files[args->ex[0]]).socket, &sock_efds);
}
}
if ((td->td_retval[0] = lwip_select(args->nd, &sock_rfds, &sock_wfds, &sock_efds, args->tv)) == -1)
error = -1;
if (args->in != 0x0) {
for (i = 0; i < MAX_FILES; i++) {
if (!FD_ISSET(((struct file * ) td->o_files[args->ou[0]]).socket, &sock_rfds))
FD_CLR(((struct file * ) td->o_files[args->ou[0]]).socket, args->in);
}
}
if (args->ou != 0x0) {
for (i = 0; i < MAX_FILES; i++) {
if (!FD_ISSET(((struct file * ) td->o_files[args->ou[0]]).socket, &sock_wfds))
FD_CLR(((struct file * ) td->o_files[args->ou[0]]).socket, args->ou);
}
}
if (args->ex != 0x0) {
for (i = 0; i < MAX_FILES; i++) {
if (!FD_ISSET(((struct file * ) td->o_files[args->ou[0]]).socket, &sock_efds))
FD_CLR(((struct file * ) td->o_files[args->ou[0]]).socket, args->ex);
}
}
*/
if ((td->td_retval[0] = lwip_select(args->nd, args->in, args->ou, args->ex, args->tv)) == -1)
error = -1;
return (error);
}
int dup2(struct thread *td, u_int32_t from, u_int32_t to) {
struct file *fp = 0x0;
struct file *dup_fp = 0x0;
if (to > MAX_FILES) {
kprintf("TO: %i > MAX_FILES: %i", to, MAX_FILES);
return (-1);
}
else if (td->o_files[to] != 0x0) {
fclose(((struct file*) td->o_files[to])->fd);
if (fdestroy(td, (struct file*) td->o_files[to], to) != 0x0)
kprintf("[%s:%i] Error with fdestroy!", __FILE__, __LINE__);
}
fp = (struct file*) td->o_files[from];
dup_fp = (struct file*) kmalloc(sizeof(struct file));
memcpy(dup_fp, fp, sizeof(struct file));
td->o_files[to] = (void*) dup_fp;
((struct file*) td->o_files[from])->fd->dup++;
return (0x0);
}
int sys_dup2(struct thread *td, struct sys_dup2_args *args) {
int error = 0x0;
if ((td->td_retval[0] = dup2(td, args->from, args->to)) == -1)
error = -1;
return (error);
}