Fix the negative offset check for pread/pwrite

This commit is contained in:
LI Qing 2020-04-28 06:26:07 +00:00 committed by Tate, Hongliang Tian
parent 48d7f8df3e
commit 6d27595195
5 changed files with 221 additions and 113 deletions

@ -12,8 +12,11 @@ pub fn do_readv(fd: FileDesc, bufs: &mut [&mut [u8]]) -> Result<usize> {
file_ref.readv(bufs)
}
pub fn do_pread(fd: FileDesc, buf: &mut [u8], offset: usize) -> Result<usize> {
pub fn do_pread(fd: FileDesc, buf: &mut [u8], offset: off_t) -> Result<usize> {
debug!("pread: fd: {}, offset: {}", fd, offset);
if offset < 0 {
return_errno!(EINVAL, "the offset is negative");
}
let file_ref = current!().file(fd)?;
file_ref.read_at(offset, buf)
file_ref.read_at(offset as usize, buf)
}

@ -12,8 +12,11 @@ pub fn do_writev(fd: FileDesc, bufs: &[&[u8]]) -> Result<usize> {
file_ref.writev(bufs)
}
pub fn do_pwrite(fd: FileDesc, buf: &[u8], offset: usize) -> Result<usize> {
pub fn do_pwrite(fd: FileDesc, buf: &[u8], offset: off_t) -> Result<usize> {
debug!("pwrite: fd: {}, offset: {}", fd, offset);
if offset < 0 {
return_errno!(EINVAL, "the offset is negative");
}
let file_ref = current!().file(fd)?;
file_ref.write_at(offset, buf)
file_ref.write_at(offset as usize, buf)
}

@ -124,7 +124,7 @@ pub fn do_readv(fd: FileDesc, iov: *mut iovec_t, count: i32) -> Result<isize> {
Ok(len as isize)
}
pub fn do_pread(fd: FileDesc, buf: *mut u8, size: usize, offset: usize) -> Result<isize> {
pub fn do_pread(fd: FileDesc, buf: *mut u8, size: usize, offset: off_t) -> Result<isize> {
let safe_buf = {
from_user::check_mut_array(buf, size)?;
unsafe { std::slice::from_raw_parts_mut(buf, size) }
@ -133,7 +133,7 @@ pub fn do_pread(fd: FileDesc, buf: *mut u8, size: usize, offset: usize) -> Resul
Ok(len as isize)
}
pub fn do_pwrite(fd: FileDesc, buf: *const u8, size: usize, offset: usize) -> Result<isize> {
pub fn do_pwrite(fd: FileDesc, buf: *const u8, size: usize, offset: off_t) -> Result<isize> {
let safe_buf = {
from_user::check_array(buf, size)?;
unsafe { std::slice::from_raw_parts(buf, size) }

@ -88,8 +88,8 @@ macro_rules! process_syscall_table_with_callback {
(RtSigprocmask = 14) => do_rt_sigprocmask(),
(RtSigreturn = 15) => handle_unsupported(),
(Ioctl = 16) => do_ioctl(fd: FileDesc, cmd: u32, argp: *mut u8),
(Pread64 = 17) => do_pread(fd: FileDesc, buf: *mut u8, size: usize, offset: usize),
(Pwrite64 = 18) => do_pwrite(fd: FileDesc, buf: *const u8, size: usize, offset: usize),
(Pread64 = 17) => do_pread(fd: FileDesc, buf: *mut u8, size: usize, offset: off_t),
(Pwrite64 = 18) => do_pwrite(fd: FileDesc, buf: *const u8, size: usize, offset: off_t),
(Readv = 19) => do_readv(fd: FileDesc, iov: *mut iovec_t, count: i32),
(Writev = 20) => do_writev(fd: FileDesc, iov: *const iovec_t, count: i32),
(Access = 21) => do_access(path: *const i8, mode: u32),

@ -1,116 +1,218 @@
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/uio.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
#include <stdio.h>
#include "test.h"
int main(int argc, const char* argv[]) {
const char* file_name = "/root/test_filesystem_file_read_write.txt";
int fd, flags, mode, len;
off_t offset;
const char* write_msg = "Hello World\n";
char read_buf[128] = {0};
// ============================================================================
// Helper function
// ============================================================================
// write
flags = O_WRONLY | O_CREAT| O_TRUNC;
mode = 00666;
if ((fd = open(file_name, flags, mode)) < 0) {
printf("ERROR: failed to open a file for write\n");
return -1;
}
if ((len = write(fd, write_msg, strlen(write_msg))) <= 0) {
printf("ERROR: failed to write to the file\n");
return -1;
}
// lseek
if ((offset = lseek(fd, 0, SEEK_END)) != 12) {
printf("ERROR: failed to lseek the file\n");
return -1;
static int create_file(const char *file_path) {
int fd;
int flags = O_RDONLY | O_CREAT| O_TRUNC;
int mode = 00666;
fd = open(file_path, flags, mode);
if (fd < 0) {
THROW_ERROR("failed to create a file");
}
close(fd);
// read
flags = O_RDONLY;
if ((fd = open(file_name, flags)) < 0) {
printf("ERROR: failed to open a file for read\n");
return -1;
}
if ((len = read(fd, read_buf, sizeof(read_buf) - 1)) <= 0) {
printf("ERROR: failed to read from the file\n");
return -1;
}
close(fd);
if (strcmp(write_msg, read_buf) != 0) {
printf("ERROR: the message read from the file is not as it was written\n");
return -1;
}
// writev
flags = O_WRONLY | O_CREAT| O_TRUNC;
if ((fd = open(file_name, flags)) < 0) {
printf("ERROR: failed to open a file for write\n");
return -1;
}
const char* iov_msg[2] = {"hello_", "world!"};
struct iovec iov[2];
for(int i=0; i<2; ++i) {
iov[i].iov_base = (void*)iov_msg[i];
iov[i].iov_len = strlen(iov_msg[i]);
}
if ((len = writev(fd, iov, 2)) != 12) {
printf("ERROR: failed to write vectors to the file\n");
return -1;
}
// pwrite
if ((len = pwrite(fd, " ", 1, 5)) != 1) {
printf("ERROR: failed to pwrite to the file\n");
}
close(fd);
flags = O_RDONLY;
if ((fd = open(file_name, flags)) < 0) {
printf("ERROR: failed to open a file for read\n");
return -1;
}
// lseek
if ((offset = lseek(fd, 2, SEEK_SET)) != 2) {
printf("ERROR: failed to lseek the file\n");
return -1;
}
// readv
iov[0].iov_base = read_buf;
iov[0].iov_len = 3;
iov[1].iov_base = read_buf + 5;
iov[1].iov_len = 20;
if ((len = readv(fd, iov, 2)) != 10) {
printf("ERROR: failed to read vectors from the file\n");
return -1;
}
if (memcmp(read_buf, "llo", 3) != 0
|| memcmp(read_buf + 5, " world!", 7) != 0) {
printf("ERROR: the message read from the file is not as it was written\n");
return -1;
}
// pread
if ((len = pread(fd, read_buf, sizeof(read_buf) - 1, 4)) != 8) {
printf("ERROR: failed to pread from the file\n");
}
if (memcmp(read_buf, "o world!", 8) != 0) {
printf("ERROR: the message read from the file is not as it was written\n");
return -1;
}
close(fd);
printf("File write and read successfully\n");
return 0;
}
static int remove_file(const char *file_path) {
int ret;
ret = unlink(file_path);
if (ret < 0) {
THROW_ERROR("failed to unlink the created file");
}
return 0;
}
// ============================================================================
// Test cases for file
// ============================================================================
static int __test_write_read(const char *file_path) {
char *write_str = "Hello World\n";
char read_buf[128] = { 0 };
int fd;
fd = open(file_path, O_WRONLY);
if (fd < 0) {
THROW_ERROR("failed to open a file to write");
}
if (write(fd, write_str, strlen(write_str)) <= 0) {
THROW_ERROR("failed to write");
}
close(fd);
fd = open(file_path, O_RDONLY);
if (fd < 0) {
THROW_ERROR("failed to open a file to read");
}
if (read(fd, read_buf, sizeof(read_buf)) != strlen(write_str)) {
THROW_ERROR("failed to read");
}
if (strcmp(write_str, read_buf) != 0) {
THROW_ERROR("the message read from the file is not as it was written");
}
close(fd);
return 0;
}
static int __test_pwrite_pread(const char *file_path) {
char *write_str = "Hello World\n";
char read_buf[128] = { 0 };
int ret, fd;
fd = open(file_path, O_WRONLY);
if (fd < 0) {
THROW_ERROR("failed to open a file to pwrite");
}
if (pwrite(fd, write_str, strlen(write_str), 1) <= 0) {
THROW_ERROR("failed to pwrite");
}
ret = pwrite(fd, write_str, strlen(write_str), -1);
if (ret >= 0 || errno != EINVAL) {
THROW_ERROR("check pwrite with negative offset fail");
}
close(fd);
fd = open(file_path, O_RDONLY);
if (fd < 0) {
THROW_ERROR("failed to open a file to pread");
}
if (pread(fd, read_buf, sizeof(read_buf), 1) != strlen(write_str)) {
THROW_ERROR("failed to pread");
}
if (strcmp(write_str, read_buf) != 0) {
THROW_ERROR("the message read from the file is not as it was written");
}
ret = pread(fd, write_str, strlen(write_str), -1);
if (ret >= 0 || errno != EINVAL) {
THROW_ERROR("check pread with negative offset fail");
}
close(fd);
return 0;
}
static int __test_writev_readv(const char *file_path) {
const char* iov_msg[2] = {"hello_", "world!"};
char read_buf[128] = { 0 };
struct iovec iov[2];
int fd, len = 0;
fd = open(file_path, O_WRONLY);
if (fd < 0) {
THROW_ERROR("failed to open a file to writev");
}
for(int i = 0; i < 2; ++i) {
iov[i].iov_base = (void*)iov_msg[i];
iov[i].iov_len = strlen(iov_msg[i]);
len += iov[i].iov_len;
}
if (writev(fd, iov, 2) != len) {
THROW_ERROR("failed to write vectors to the file");
return -1;
}
close(fd);
fd = open(file_path, O_RDONLY);
if (fd < 0) {
THROW_ERROR("failed to open a file to readv");
}
iov[0].iov_base = read_buf;
iov[0].iov_len = strlen(iov_msg[0]);
iov[1].iov_base = read_buf + strlen(iov_msg[0]);
iov[1].iov_len = strlen(iov_msg[1]);
if (readv(fd, iov, 2) != len) {
THROW_ERROR("failed to read vectors from the file");
}
if (memcmp(read_buf, iov_msg[0], strlen(iov_msg[0])) != 0 ||
memcmp(read_buf + strlen(iov_msg[0]), iov_msg[1], strlen(iov_msg[1])) != 0) {
THROW_ERROR("the message read from the file is not as it was written");
}
close(fd);
return 0;
}
static int __test_lseek(const char *file_path) {
char *write_str = "Hello World\n";
char read_buf[128] = { 0 };
int fd, offset, ret;
fd = open(file_path, O_RDWR);
if (fd < 0) {
THROW_ERROR("failed to open a file to read/write");
}
if (write(fd, write_str, strlen(write_str)) <= 0) {
THROW_ERROR("failed to write");
}
/* make sure offset is in range (0, strlen(write_str)) */
offset = 2;
if (lseek(fd, offset, SEEK_SET) != offset) {
THROW_ERROR("failed to lseek the file");
}
if (read(fd, read_buf, sizeof(read_buf)) >= strlen(write_str)) {
THROW_ERROR("failed to read from offset");
}
if (strcmp(write_str + offset, read_buf) != 0) {
THROW_ERROR("the message read from the offset is wrong");
}
offset = -1;
ret = lseek(fd, offset, SEEK_SET);
if (ret >= 0 || errno != EINVAL) {
THROW_ERROR("check lseek with negative offset fail");
}
if (lseek(fd, 0, SEEK_END) != strlen(write_str)) {
THROW_ERROR("faild to lseek to the end of the file");
}
close(fd);
return 0;
}
typedef int(*test_file_func_t)(const char *);
static int test_file_framework(test_file_func_t fn) {
const char *file_path = "/root/test_filesystem_file_read_write.txt";
if (create_file(file_path) < 0)
return -1;
if (fn(file_path) < 0)
return -1;
if (remove_file(file_path) < 0)
return -1;
return 0;
}
static int test_write_read() {
return test_file_framework(__test_write_read);
}
static int test_pwrite_pread() {
return test_file_framework(__test_pwrite_pread);
}
static int test_writev_readv() {
return test_file_framework(__test_writev_readv);
}
static int test_lseek() {
return test_file_framework(__test_lseek);
}
// ============================================================================
// Test suite main
// ============================================================================
static test_case_t test_cases[] = {
TEST_CASE(test_write_read),
TEST_CASE(test_pwrite_pread),
TEST_CASE(test_writev_readv),
TEST_CASE(test_lseek),
};
int main(int argc, const char *argv[]) {
return test_suite_run(test_cases, ARRAY_SIZE(test_cases));
}