mirror of
https://github.com/apache/nuttx.git
synced 2025-01-13 10:58:49 +08:00
fdcheck: Enable fdcheck to automatically detect ownership of fd
Signed-off-by: hujun5 <hujun5@xiaomi.com>
This commit is contained in:
parent
4b5ad956c3
commit
577eb47966
8 changed files with 137 additions and 39 deletions
|
@ -254,7 +254,11 @@ static int nx_dup3_from_tcb(FAR struct tcb_s *tcb, int fd1, int fd2,
|
||||||
ret = file_dup3(files_fget(list, fd1), filep, flags);
|
ret = file_dup3(files_fget(list, fd1), filep, flags);
|
||||||
|
|
||||||
#ifdef CONFIG_FDSAN
|
#ifdef CONFIG_FDSAN
|
||||||
filep->f_tag = file.f_tag;
|
filep->f_tag_fdsan = file.f_tag_fdsan;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDCHECK
|
||||||
|
filep->f_tag_fdcheck = file.f_tag_fdcheck;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
file_close(&file);
|
file_close(&file);
|
||||||
|
|
|
@ -163,6 +163,16 @@ int file_dup3(FAR struct file *filep1, FAR struct file *filep2, int flags)
|
||||||
ret = file_close(filep2);
|
ret = file_close(filep2);
|
||||||
DEBUGASSERT(ret == 0);
|
DEBUGASSERT(ret == 0);
|
||||||
|
|
||||||
|
/* Copy tag */
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDSAN
|
||||||
|
temp.f_tag_fdsan = filep1->f_tag_fdsan;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDCHECK
|
||||||
|
temp.f_tag_fdcheck = filep1->f_tag_fdcheck;
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Return the file structure */
|
/* Return the file structure */
|
||||||
|
|
||||||
memcpy(filep2, &temp, sizeof(temp));
|
memcpy(filep2, &temp, sizeof(temp));
|
||||||
|
|
|
@ -43,9 +43,6 @@
|
||||||
static int file_vioctl(FAR struct file *filep, int req, va_list ap)
|
static int file_vioctl(FAR struct file *filep, int req, va_list ap)
|
||||||
{
|
{
|
||||||
FAR struct inode *inode;
|
FAR struct inode *inode;
|
||||||
#ifdef CONFIG_FDSAN
|
|
||||||
FAR uint64_t *tag;
|
|
||||||
#endif
|
|
||||||
unsigned long arg;
|
unsigned long arg;
|
||||||
int ret = -ENOTTY;
|
int ret = -ENOTTY;
|
||||||
|
|
||||||
|
@ -113,15 +110,25 @@ static int file_vioctl(FAR struct file *filep, int req, va_list ap)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
#ifdef CONFIG_FDSAN
|
#ifdef CONFIG_FDSAN
|
||||||
case FIOC_SETTAG:
|
case FIOC_SETTAG_FDSAN:
|
||||||
tag = (FAR uint64_t *)arg;
|
filep->f_tag_fdsan = *(FAR uint64_t *)arg;
|
||||||
filep->f_tag = *tag;
|
|
||||||
ret = OK;
|
ret = OK;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FIOC_GETTAG:
|
case FIOC_GETTAG_FDSAN:
|
||||||
tag = (FAR uint64_t *)arg;
|
*(FAR uint64_t *)arg = filep->f_tag_fdsan;
|
||||||
*tag = filep->f_tag;
|
ret = OK;
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDCHECK
|
||||||
|
case FIOC_SETTAG_FDCHECK:
|
||||||
|
filep->f_tag_fdcheck = *(FAR uint8_t *)arg;
|
||||||
|
ret = OK;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case FIOC_GETTAG_FDCHECK:
|
||||||
|
*(FAR uint8_t *)arg = filep->f_tag_fdcheck;
|
||||||
ret = OK;
|
ret = OK;
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -46,21 +46,22 @@ extern "C"
|
||||||
*
|
*
|
||||||
* Description: Obtain original fd information
|
* Description: Obtain original fd information
|
||||||
*
|
*
|
||||||
* Val carries the pid and fd information.
|
* Val carries the pid, tag and fd information.
|
||||||
* The original fd information is stored in low bit of val.
|
* The original fd information is stored in low bit of val.
|
||||||
* The pid information is stored in the high bit of val.
|
* The pid and tag information is stored in the high bit of val.
|
||||||
* For ease of understanding, let's give an example where
|
* For ease of understanding, let's give an example where
|
||||||
* the following information is represented in 32-bit binary format
|
* the following information is represented in 32-bit binary format
|
||||||
*
|
*
|
||||||
* val 00000000 00000000 01010101 10001010
|
* val 00000000 01010101 00000001 10001010
|
||||||
* fd 00000000 00000000 00000000 10001010
|
* fd 00000000 00000000 00000000 10001010
|
||||||
* pid 00000000 00000000 00000000 01010101
|
* pid 00000000 00000000 00000000 01010101
|
||||||
|
* tag 00000000 00000000 00000000 00000001
|
||||||
*
|
*
|
||||||
* In this function, we also check if the pid information is correct.
|
* In this function, we also check if the pid and tag information is correct.
|
||||||
* If there is an error, it will panic.
|
* If there is an error, it will panic.
|
||||||
*
|
*
|
||||||
* Input Parameters:
|
* Input Parameters:
|
||||||
* val - this val carrying pid and original fd information
|
* val - this val carrying pid, tag and original fd information
|
||||||
*
|
*
|
||||||
* Returned Value: none
|
* Returned Value: none
|
||||||
*
|
*
|
||||||
|
@ -71,17 +72,18 @@ int fdcheck_restore(int fd);
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: fdcheck_protect
|
* Name: fdcheck_protect
|
||||||
*
|
*
|
||||||
* Description: Obtain the combined value of fd and pid
|
* Description: Obtain the combined value of fd, pid and tag
|
||||||
*
|
*
|
||||||
* the return value carries the pid and fd information.
|
* the return value carries the pid, tag and fd information.
|
||||||
* The original fd information is stored in low bit of val.
|
* The original fd information is stored in low bit of val.
|
||||||
* The pid information is stored in high bit of val.
|
* The pid and tag information is stored in high bit of val.
|
||||||
* For ease of understanding, let's give an example where
|
* For ease of understanding, let's give an example where
|
||||||
* the following information is represented in 32-bit binary format
|
* the following information is represented in 32-bit binary format
|
||||||
*
|
*
|
||||||
* fd 00000000 00000000 00000000 10001010
|
* fd 00000000 00000000 00000000 10001010
|
||||||
* pid 00000000 00000000 00000000 01010101
|
* pid 00000000 00000000 00000000 01010101
|
||||||
* val 00000000 00000000 01010101 10001010
|
* tag 00000000 00000000 00000000 00000001
|
||||||
|
* val 00000000 01010101 00000001 10001010
|
||||||
*
|
*
|
||||||
* Input Parameters:
|
* Input Parameters:
|
||||||
* fd - original fd
|
* fd - original fd
|
||||||
|
|
|
@ -468,7 +468,11 @@ struct file
|
||||||
FAR struct inode *f_inode; /* Driver or file system interface */
|
FAR struct inode *f_inode; /* Driver or file system interface */
|
||||||
FAR void *f_priv; /* Per file driver private data */
|
FAR void *f_priv; /* Per file driver private data */
|
||||||
#ifdef CONFIG_FDSAN
|
#ifdef CONFIG_FDSAN
|
||||||
uint64_t f_tag; /* file owner tag, init to 0 */
|
uint64_t f_tag_fdsan; /* File owner fdsan tag, init to 0 */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDCHECK
|
||||||
|
uint8_t f_tag_fdcheck; /* File owner fdcheck tag, init to 0 */
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -189,17 +189,29 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifdef CONFIG_FDSAN
|
#ifdef CONFIG_FDSAN
|
||||||
#define FIOC_SETTAG _FIOC(0x000e) /* IN: FAR uint64_t *
|
#define FIOC_SETTAG_FDSAN _FIOC(0x000e) /* IN: FAR uint64_t *
|
||||||
* Pointer to file tag
|
* Pointer to file tag
|
||||||
* OUT: None
|
* OUT: None
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define FIOC_GETTAG _FIOC(0x000f) /* IN: FAR uint64_t *
|
#define FIOC_GETTAG_FDSAN _FIOC(0x000f) /* IN: FAR uint64_t *
|
||||||
* Pointer to file tag
|
* Pointer to file tag
|
||||||
* OUT: None
|
* OUT: None
|
||||||
*/
|
*/
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDCHECK
|
||||||
|
#define FIOC_SETTAG_FDCHECK _FIOC(0x0010) /* IN: FAR uint8_t *
|
||||||
|
* Pointer to file fdcheck tag
|
||||||
|
* OUT: None
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define FIOC_GETTAG_FDCHECK _FIOC(0x0011) /* IN: FAR uint8_t *
|
||||||
|
* Pointer to file fdcheck tag
|
||||||
|
* OUT: None
|
||||||
|
*/
|
||||||
|
#endif
|
||||||
|
|
||||||
/* NuttX file system ioctl definitions **************************************/
|
/* NuttX file system ioctl definitions **************************************/
|
||||||
|
|
||||||
#define _DIOCVALID(c) (_IOC_TYPE(c)==_DIOCBASE)
|
#define _DIOCVALID(c) (_IOC_TYPE(c)==_DIOCBASE)
|
||||||
|
|
|
@ -25,6 +25,8 @@
|
||||||
#include <nuttx/fdcheck.h>
|
#include <nuttx/fdcheck.h>
|
||||||
#include <nuttx/lib/math32.h>
|
#include <nuttx/lib/math32.h>
|
||||||
#include <nuttx/sched.h>
|
#include <nuttx/sched.h>
|
||||||
|
#include <nuttx/spinlock.h>
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
|
||||||
#include <debug.h>
|
#include <debug.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
@ -39,10 +41,21 @@
|
||||||
#define FD_BITS LOG2_CEIL(OPEN_MAX)
|
#define FD_BITS LOG2_CEIL(OPEN_MAX)
|
||||||
#define FD_MASK ((1 << FD_BITS) - 1)
|
#define FD_MASK ((1 << FD_BITS) - 1)
|
||||||
|
|
||||||
#define PID_SHIFT (FD_BITS + FD_SHIFT)
|
#define TAG_SHIFT (FD_BITS + FD_SHIFT)
|
||||||
|
#define TAG_BITS 8
|
||||||
|
#define TAG_MASK ((1 << TAG_BITS) - 1)
|
||||||
|
|
||||||
|
#define PID_SHIFT (TAG_BITS + TAG_SHIFT)
|
||||||
#define PID_BITS (8 * sizeof(int) - 1 - PID_SHIFT)
|
#define PID_BITS (8 * sizeof(int) - 1 - PID_SHIFT)
|
||||||
#define PID_MASK ((1 << PID_BITS) - 1)
|
#define PID_MASK ((1 << PID_BITS) - 1)
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Private Data
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
static spinlock_t g_fdcheck_lock = SP_UNLOCKED;
|
||||||
|
static uint8_t g_fdcheck_tag = 0;
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Public Functions
|
* Public Functions
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
@ -52,21 +65,22 @@
|
||||||
*
|
*
|
||||||
* Description: Obtain original fd information
|
* Description: Obtain original fd information
|
||||||
*
|
*
|
||||||
* Val carries the pid and fd information.
|
* Val carries the pid, tag and fd information.
|
||||||
* The original fd information is stored in low bit of val.
|
* The original fd information is stored in low bit of val.
|
||||||
* The pid information is stored in the high bit of val.
|
* The pid and tag information is stored in the high bit of val.
|
||||||
* For ease of understanding, let's give an example where
|
* For ease of understanding, let's give an example where
|
||||||
* the following information is represented in 32-bit binary format
|
* the following information is represented in 32-bit binary format
|
||||||
*
|
*
|
||||||
* val 00000000 00000000 01010101 10001010
|
* val 00000000 01010101 00000001 10001010
|
||||||
* fd 00000000 00000000 00000000 10001010
|
* fd 00000000 00000000 00000000 10001010
|
||||||
* pid 00000000 00000000 00000000 01010101
|
* pid 00000000 00000000 00000000 01010101
|
||||||
|
* tag 00000000 00000000 00000000 00000001
|
||||||
*
|
*
|
||||||
* In this function, we also check if the pid information is correct.
|
* In this function, we also check if the pid and tag information is correct.
|
||||||
* If there is an error, it will panic.
|
* If there is an error, it will panic.
|
||||||
*
|
*
|
||||||
* Input Parameters:
|
* Input Parameters:
|
||||||
* val - this val carrying pid and original fd information
|
* val - this val carrying pid, tag and original fd information
|
||||||
*
|
*
|
||||||
* Returned Value: none
|
* Returned Value: none
|
||||||
*
|
*
|
||||||
|
@ -75,17 +89,17 @@
|
||||||
int fdcheck_restore(int val)
|
int fdcheck_restore(int val)
|
||||||
{
|
{
|
||||||
int pid_expect;
|
int pid_expect;
|
||||||
int pid_now;
|
|
||||||
int ppid_now;
|
int ppid_now;
|
||||||
|
int pid_now;
|
||||||
|
|
||||||
if (val <= 2)
|
if (val <= 2)
|
||||||
{
|
{
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
pid_expect = (val >> PID_SHIFT);
|
pid_expect = (val >> PID_SHIFT) & PID_MASK;
|
||||||
pid_now = (_SCHED_GETPID() & PID_MASK);
|
pid_now = _SCHED_GETPID() & PID_MASK;
|
||||||
ppid_now = (_SCHED_GETPPID() & PID_MASK);
|
ppid_now = _SCHED_GETPPID() & PID_MASK;
|
||||||
if (pid_expect != pid_now && pid_expect != ppid_now && pid_expect != 0)
|
if (pid_expect != pid_now && pid_expect != ppid_now && pid_expect != 0)
|
||||||
{
|
{
|
||||||
ferr("pid_expect %d pid_now %d ppid_now %d\n",
|
ferr("pid_expect %d pid_now %d ppid_now %d\n",
|
||||||
|
@ -93,23 +107,40 @@ int fdcheck_restore(int val)
|
||||||
PANIC();
|
PANIC();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (pid_expect != 0)
|
||||||
|
{
|
||||||
|
uint8_t tag_store;
|
||||||
|
int ret = ioctl(val & FD_MASK, FIOC_GETTAG_FDCHECK, &tag_store);
|
||||||
|
if (ret >= 0)
|
||||||
|
{
|
||||||
|
uint8_t tag_expect = (val >> TAG_SHIFT) & TAG_MASK;
|
||||||
|
if (tag_expect != tag_store)
|
||||||
|
{
|
||||||
|
ferr("tag_expect 0x%x tag_store 0x%x\n",
|
||||||
|
tag_expect, tag_store);
|
||||||
|
PANIC();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return val & FD_MASK;
|
return val & FD_MASK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: fdcheck_protect
|
* Name: fdcheck_protect
|
||||||
*
|
*
|
||||||
* Description: Obtain the combined value of fd and pid
|
* Description: Obtain the combined value of fd, pid and tag
|
||||||
*
|
*
|
||||||
* the return value carries the pid and fd information.
|
* the return value carries the pid, tag and fd information.
|
||||||
* The original fd information is stored in low bit of val.
|
* The original fd information is stored in low bit of val.
|
||||||
* The pid information is stored in high bit of val.
|
* The pid and tag information is stored in high bit of val.
|
||||||
* For ease of understanding, let's give an example where
|
* For ease of understanding, let's give an example where
|
||||||
* the following information is represented in 32-bit binary format
|
* the following information is represented in 32-bit binary format
|
||||||
*
|
*
|
||||||
* fd 00000000 00000000 00000000 10001010
|
* fd 00000000 00000000 00000000 10001010
|
||||||
* pid 00000000 00000000 00000000 01010101
|
* pid 00000000 00000000 00000000 01010101
|
||||||
* val 00000000 00000000 01010101 10001010
|
* tag 00000000 00000000 00000000 00000001
|
||||||
|
* val 00000000 01010101 00000001 10001010
|
||||||
*
|
*
|
||||||
* Input Parameters:
|
* Input Parameters:
|
||||||
* fd - original fd
|
* fd - original fd
|
||||||
|
@ -120,12 +151,40 @@ int fdcheck_restore(int val)
|
||||||
|
|
||||||
int fdcheck_protect(int fd)
|
int fdcheck_protect(int fd)
|
||||||
{
|
{
|
||||||
|
int protect_fd;
|
||||||
|
uint8_t tag;
|
||||||
|
int ret;
|
||||||
|
|
||||||
if (fd <= 2)
|
if (fd <= 2)
|
||||||
{
|
{
|
||||||
return fd;
|
return fd;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (fd & FD_MASK) | ((_SCHED_GETPID() & PID_MASK) << PID_SHIFT);
|
protect_fd = fd & FD_MASK;
|
||||||
|
protect_fd |= (_SCHED_GETPID() & PID_MASK) << PID_SHIFT;
|
||||||
|
|
||||||
|
ret = ioctl(fd, FIOC_GETTAG_FDCHECK, &tag);
|
||||||
|
DEBUGASSERT(ret >= 0);
|
||||||
|
if (tag == 0)
|
||||||
|
{
|
||||||
|
irqstate_t flags = spin_lock_irqsave(&g_fdcheck_lock);
|
||||||
|
if ((++g_fdcheck_tag & TAG_MASK) == 0)
|
||||||
|
{
|
||||||
|
++g_fdcheck_tag;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_fdcheck_tag &= TAG_MASK;
|
||||||
|
protect_fd |= g_fdcheck_tag << TAG_SHIFT;
|
||||||
|
ret = ioctl(fd, FIOC_SETTAG_FDCHECK, &g_fdcheck_tag);
|
||||||
|
DEBUGASSERT(ret == 0);
|
||||||
|
spin_unlock_irqrestore(&g_fdcheck_lock, flags);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
protect_fd |= (tag & TAG_MASK) << TAG_SHIFT;
|
||||||
|
}
|
||||||
|
|
||||||
|
return protect_fd;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -151,7 +151,7 @@ void android_fdsan_exchange_owner_tag(int fd, uint64_t expected_tag,
|
||||||
uint64_t tag;
|
uint64_t tag;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = ioctl(fd, FIOC_GETTAG, &tag);
|
ret = ioctl(fd, FIOC_GETTAG_FDSAN, &tag);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
|
@ -159,7 +159,7 @@ void android_fdsan_exchange_owner_tag(int fd, uint64_t expected_tag,
|
||||||
|
|
||||||
if (tag == expected_tag)
|
if (tag == expected_tag)
|
||||||
{
|
{
|
||||||
ret = ioctl(fd, FIOC_SETTAG, &new_tag);
|
ret = ioctl(fd, FIOC_SETTAG_FDSAN, &new_tag);
|
||||||
DEBUGASSERT(ret == 0);
|
DEBUGASSERT(ret == 0);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
Loading…
Reference in a new issue