diff options
Diffstat (limited to 'libgloss/mips/uhi/yamon_write.c')
-rw-r--r-- | libgloss/mips/uhi/yamon_write.c | 98 |
1 files changed, 98 insertions, 0 deletions
diff --git a/libgloss/mips/uhi/yamon_write.c b/libgloss/mips/uhi/yamon_write.c new file mode 100644 index 0000000..f42bb65 --- /dev/null +++ b/libgloss/mips/uhi/yamon_write.c @@ -0,0 +1,98 @@ +/* + * Copyright (C) 2014-2018 MIPS Tech, LLC + * + * 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. Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * 3. Neither the name of the copyright holder 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 COPYRIGHT HOLDERS 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 HOLDER 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. +*/ + +/* + * @Synopsis int32_t write (int32_t fd, void *buffer, int32_t count); + * + * Parameters: + * $4 - File handle + * $5 - Buffer to write + * $6 - Length of the buffer + * + * Return: + * $2 - Number of bytes written + * + * Arguments to syscall: + * $25 - Operation code for write + * $4 - File handle + * $5 - Buffer to write + * $6 - Length of the buffer + * + * Return from syscall: + * $2 - Number of bytes written + * $3 - errno + * + * @Description File write +*/ + +#include <stdint.h> +#include <errno.h> +#include "yamon_syscalls.h" +#include <mips/uhi_syscalls.h> +#include <mips/hal.h> + +int32_t _MIPS_HAL_NOMIPS16 +write (int32_t fd, void *buffer, int32_t count) +{ + register int32_t arg1 asm ("$4") = fd; + register void *arg2 asm ("$5") = buffer; + register int32_t arg3 asm ("$6") = count; + register int32_t op asm ("$25") = __MIPS_UHI_WRITE; + register int32_t ret asm ("$2") = __MIPS_UHI_SYSCALL_NUM; + register int32_t new_errno asm ("$3") = 0; + + /* yamon print_count always writes to stdout */ + if (fd == 1 || fd == 2) + { + typedef int32_t (*funcptr) (int32_t, char *, int32_t); + extern funcptr __yamon_functions[]; + funcptr yamonwrite; + + yamonwrite = (funcptr)__yamon_functions[__YAMON_PRINT_COUNT]; + + ret = (yamonwrite) (0, buffer, count); + } + else + { + /* Use UHI write when writing to destination other than stdout */ + __asm__ __volatile__(" # %0,%1 = write(%2, %3, %4) op=%5\n" + SYSCALL (__MIPS_UHI_SYSCALL_NUM) + : "+r" (ret), "=r" (new_errno), "+r" (arg1), "+r" (arg2) + : "r" (arg3), "r" (op)); + } + if (ret < 0) + { + /* Do a dance to set errno, errno is a function call that can + clobber $3. */ + volatile uint32_t errno_tmp = new_errno; + errno = errno_tmp; + } + + return ret; +} + |