2018-07-19 08:42:12 +01:00
|
|
|
#include "devoptab_fs.h"
|
2018-05-27 12:11:45 +01:00
|
|
|
|
|
|
|
ssize_t
|
|
|
|
__wut_fs_write(struct _reent *r,
|
|
|
|
void *fd,
|
|
|
|
const char *ptr,
|
|
|
|
size_t len)
|
|
|
|
{
|
2019-01-23 19:30:07 +01:00
|
|
|
FSStatus status = 0;
|
2018-10-06 10:41:24 +01:00
|
|
|
FSCmdBlock cmd;
|
|
|
|
uint8_t *alignedWriteBuffer;
|
|
|
|
uint32_t bytes, bytesWritten;
|
|
|
|
__wut_fs_file_t *file;
|
|
|
|
|
|
|
|
if (!fd || !ptr) {
|
|
|
|
r->_errno = EINVAL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
FSInitCmdBlock(&cmd);
|
|
|
|
file = (__wut_fs_file_t *)fd;
|
|
|
|
bytesWritten = 0;
|
2018-05-27 12:11:45 +01:00
|
|
|
|
|
|
|
// Check that the file was opened with write access
|
|
|
|
if ((file->flags & O_ACCMODE) == O_RDONLY) {
|
|
|
|
r->_errno = EBADF;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2021-01-01 17:16:40 +01:00
|
|
|
if((((uintptr_t) ptr) & 0x3F) == 0){
|
|
|
|
status = FSWriteFile(__wut_devoptab_fs_client, &cmd, (uint8_t *) ptr,
|
2020-12-29 09:37:32 +01:00
|
|
|
1, len, file->fd, 0, FS_ERROR_FLAG_ALL);
|
2021-01-01 17:16:40 +01:00
|
|
|
if(status > 0){
|
|
|
|
bytesWritten = (uint32_t) status;
|
|
|
|
file->offset += bytesWritten;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Copy to internal buffer due to alignment requirement and read in chunks.
|
|
|
|
// Using a buffer smaller than 128KiB takes a performance hit.
|
|
|
|
int buffer_size = len < 128*1024 ? len : 128*1024;
|
|
|
|
alignedWriteBuffer = memalign(0x40, buffer_size);
|
|
|
|
if(!alignedWriteBuffer){
|
|
|
|
r->_errno = ENOMEM;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
while (len > 0) {
|
|
|
|
size_t toWrite = len > buffer_size ? buffer_size : len;
|
2018-05-27 12:11:45 +01:00
|
|
|
|
2021-01-01 17:16:40 +01:00
|
|
|
// Copy to internal buffer
|
|
|
|
memcpy(alignedWriteBuffer, ptr, toWrite);
|
2018-05-27 12:11:45 +01:00
|
|
|
|
2021-01-01 17:16:40 +01:00
|
|
|
// Write the data
|
|
|
|
status = FSWriteFile(__wut_devoptab_fs_client, &cmd, alignedWriteBuffer,
|
2020-12-29 09:37:32 +01:00
|
|
|
1, toWrite, file->fd, 0, FS_ERROR_FLAG_ALL);
|
2021-01-01 17:16:40 +01:00
|
|
|
if (status <= 0) {
|
|
|
|
break;
|
|
|
|
}
|
2018-05-27 12:11:45 +01:00
|
|
|
|
2021-01-01 17:16:40 +01:00
|
|
|
bytes = (uint32_t)status;
|
|
|
|
file->offset += bytes;
|
|
|
|
bytesWritten += bytes;
|
|
|
|
ptr += bytes;
|
|
|
|
len -= bytes;
|
2018-10-06 10:41:24 +01:00
|
|
|
|
2021-01-01 17:16:40 +01:00
|
|
|
if (bytes < toWrite) {
|
|
|
|
break;
|
|
|
|
}
|
2018-10-06 10:41:24 +01:00
|
|
|
}
|
2021-01-01 17:16:40 +01:00
|
|
|
free(alignedWriteBuffer);
|
2018-10-06 10:41:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Return partial write
|
|
|
|
if (bytesWritten > 0) {
|
|
|
|
return bytesWritten;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status < 0) {
|
|
|
|
r->_errno = __wut_fs_translate_error(status);
|
|
|
|
return -1;
|
2018-05-27 12:11:45 +01:00
|
|
|
}
|
|
|
|
|
2018-10-06 10:41:24 +01:00
|
|
|
return 0;
|
2018-05-27 12:11:45 +01:00
|
|
|
}
|