Skip to content

Commit

Permalink
libc/misc: add fdsan module
Browse files Browse the repository at this point in the history
FD (file descriptor) is widely used in system software development,
and almost all implementations of posix os (including nuttx) use FD as an index.
the value of fd needs to be allocated starting from the minimum available value of 3, and each process has a copy,
so the same fd value is very easy to reuse in the program.

In multi threaded or multi process environments without address isolation,
If the ownership, global variables, and competition relationships of fd are not properly handled,
there may be issues with fd duplication or accidental closure.
Further leading to the following issues, which are difficult to troubleshoot.

1. Security vulnerability: the fd we wrote is not the expected fd and will be accessed by hackers to obtain data
2. Program exceptions or crashes: write or read fd failures, and program logic errors
3. The structured file XML or database is damaged: the data format written to the database is not the expected format.

The implementation principle of fdsan is based on the implementation of Android
https://android.googlesource.com/platform/bionic/+/master/docs/fdsan.md

Signed-off-by: hujun5 <hujun5@xiaomi.com>
  • Loading branch information
hujun260 authored and xiaoxiang781216 committed May 17, 2023
1 parent 2421275 commit 90387a5
Show file tree
Hide file tree
Showing 12 changed files with 496 additions and 0 deletions.
8 changes: 8 additions & 0 deletions fs/inode/fs_files.c
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,10 @@
#include <nuttx/mutex.h>
#include <nuttx/sched.h>

#ifdef CONFIG_FDSAN
# include <android/fdsan.h>
#endif

#include "inode/inode.h"

/****************************************************************************
Expand Down Expand Up @@ -686,6 +690,10 @@ int close(int fd)
{
int ret;

#ifdef CONFIG_FDSAN
android_fdsan_exchange_owner_tag(fd, 0, 0);
#endif

/* close() is a cancellation point */

enter_cancellation_point();
Expand Down
17 changes: 17 additions & 0 deletions fs/vfs/fs_ioctl.c
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,9 @@
int file_vioctl(FAR struct file *filep, int req, va_list ap)
{
FAR struct inode *inode;
#ifdef CONFIG_FDSAN
FAR uint64_t *tag;
#endif
unsigned long arg;
int ret = -ENOTTY;

Expand Down Expand Up @@ -109,6 +112,20 @@ int file_vioctl(FAR struct file *filep, int req, va_list ap)
}
break;

#ifdef CONFIG_FDSAN
case FIOC_SETTAG:
tag = (FAR uint64_t *)arg;
filep->f_tag = *tag;
ret = OK;
break;

case FIOC_GETTAG:
tag = (FAR uint64_t *)arg;
*tag = filep->f_tag;
ret = OK;
break;
#endif

#ifndef CONFIG_DISABLE_MOUNTPOINT
case BIOC_BLKSSZGET:
if (ret == -ENOTTY && inode->u.i_ops != NULL &&
Expand Down
190 changes: 190 additions & 0 deletions include/android/fdsan.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,190 @@
/****************************************************************************
* include/android/fdsan.h
* Copyright (C) 2018 The Android Open Source Project
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * 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.
*
* 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 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.
****************************************************************************/

#ifndef __INCLUDE_ANDROID_FDSAN_H
#define __INCLUDE_ANDROID_FDSAN_H

/****************************************************************************
* Included Files
****************************************************************************/

#include <stdbool.h>
#include <stdint.h>

/****************************************************************************
* Pre-processor Definitions
****************************************************************************/

/****************************************************************************
* Error checking for close(2).
*
* Mishandling of file descriptor ownership is a common source of errors that
* can be extremely difficult to diagnose. Mistakes like the following can
* result in seemingly 'impossible' failures showing up on other threads that
* happened to try to open a file descriptor
* between the buggy code's close and fclose:
*
* int print(int fd) {
* int rc;
* char buf[128];
* while ((rc = read(fd, buf, sizeof(buf))) > 0) {
* printf("%.*s", rc);
* }
* close(fd);
* }
*
* int bug() {
* FILE* f = fopen("foo", "r");
* print(fileno(f));
* fclose(f);
* }
*
* To make it easier to find this class of bugs, bionic provides a method to
* require that file descriptors are closed by their owners. File descriptors
* can be associated with tags with which they must be closed. This allows
* objects that conceptually own an fd (FILE*, unique_fd, etc.) to use their
* own address at the tag, to enforce that closure of the fd must come as a
* result of their own destruction (fclose, ~unique_fd, etc.)
*
* By default, a file descriptor's tag is 0, and close(fd) is equivalent to
* closing fd with the tag 0.
****************************************************************************/

/****************************************************************************
* For improved diagnostics, the type of a file descriptors owner can be
* encoded in the most significant byte of the owner tag. Values of 0 and
* 0xff are ignored, which allows for raw pointers to be used as owner tags
* without modification.
****************************************************************************/

#ifdef __cplusplus
#define EXTERN extern "C"
extern "C"
{
#else
#define EXTERN extern
#endif

enum android_fdsan_owner_type
{
/**************************************************************************
* Generic Java or native owners.
*
* Generic Java objects always use 255 as their type, using
* identityHashCode as the value of the tag, leaving bits 33-56 unset.
* Native pointers are sign extended from 48-bits of virtual address space,
* and so can have the MSB set to 255 as well. Use the value of bits 49-56
* to distinguish between these cases.
**************************************************************************/

ANDROID_FDSAN_OWNER_TYPE_GENERIC_00 = 0,
ANDROID_FDSAN_OWNER_TYPE_GENERIC_FF = 255,

/* FILE* */

ANDROID_FDSAN_OWNER_TYPE_FILE = 1,

/* DIR* */

ANDROID_FDSAN_OWNER_TYPE_DIR = 2,

/* android::base::unique_fd */

ANDROID_FDSAN_OWNER_TYPE_UNIQUE_FD = 3,

/* sqlite-owned file descriptors */

ANDROID_FDSAN_OWNER_TYPE_SQLITE = 4,

/* java.io.FileInputStream */

ANDROID_FDSAN_OWNER_TYPE_FILEINPUTSTREAM = 5,

/* java.io.FileOutputStream */

ANDROID_FDSAN_OWNER_TYPE_FILEOUTPUTSTREAM = 6,

/* java.io.RandomAccessFile */

ANDROID_FDSAN_OWNER_TYPE_RANDOMACCESSFILE = 7,

/* android.os.ParcelFileDescriptor */

ANDROID_FDSAN_OWNER_TYPE_PARCELFILEDESCRIPTOR = 8,

/* ART FdFile */

ANDROID_FDSAN_OWNER_TYPE_ART_FDFILE = 9,

/* java.net.DatagramSocketImpl */

ANDROID_FDSAN_OWNER_TYPE_DATAGRAMSOCKETIMPL = 10,

/* java.net.SocketImpl */

ANDROID_FDSAN_OWNER_TYPE_SOCKETIMPL = 11,

/* libziparchive's ZipArchive */

ANDROID_FDSAN_OWNER_TYPE_ZIPARCHIVE = 12,
};

typedef enum android_fdsan_owner_type android_fdsan_owner_type_t;

/****************************************************************************
* Create an owner tag with the specified type and
* least significant 56 bits of tag.
****************************************************************************/

uint64_t android_fdsan_create_owner_tag(enum android_fdsan_owner_type type,
uint64_t tag);

/****************************************************************************
* Exchange a file descriptor's tag.
*
* Logs and aborts if the fd's tag does not match expected_tag.
****************************************************************************/

void android_fdsan_exchange_owner_tag(int fd, uint64_t expected_tag,
uint64_t new_tag);

/****************************************************************************
* Close a file descriptor with a tag, and resets the tag to 0.
*
* Logs and aborts if the tag is incorrect.
****************************************************************************/

int android_fdsan_close_with_tag(int fd, uint64_t tag);

#undef EXTERN
#ifdef __cplusplus
}
#endif

#endif /* __INCLUDE_ANDROID_FDSAN_H */
3 changes: 3 additions & 0 deletions include/nuttx/fs/fs.h
Original file line number Diff line number Diff line change
Expand Up @@ -430,6 +430,9 @@ struct file
off_t f_pos; /* File position */
FAR struct inode *f_inode; /* Driver or file system interface */
FAR void *f_priv; /* Per file driver private data */
#ifdef CONFIG_FDSAN
uint64_t f_tag; /* file owner tag, init to 0 */
#endif
};

/* This defines a two layer array of files indexed by the file descriptor.
Expand Down
12 changes: 12 additions & 0 deletions include/nuttx/fs/ioctl.h
Original file line number Diff line number Diff line change
Expand Up @@ -184,6 +184,18 @@
* OUT: None
*/

#ifdef CONFIG_FDSAN
#define FIOC_SETTAG _FIOC(0x000e) /* IN: FAR uint64_t *
* Pointer to file tag
* OUT: None
*/

#define FIOC_GETTAG _FIOC(0x000f) /* IN: FAR uint64_t *
* Pointer to file tag
* OUT: None
*/
#endif

/* NuttX file system ioctl definitions **************************************/

#define _DIOCVALID(c) (_IOC_TYPE(c)==_DIOCBASE)
Expand Down
15 changes: 15 additions & 0 deletions libs/libc/dirent/lib_closedir.c
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,11 @@

#include <dirent.h>
#include <errno.h>

#ifdef CONFIG_FDSAN
# include <android/fdsan.h>
#endif

#include <unistd.h>

#include "libc.h"
Expand Down Expand Up @@ -56,14 +61,24 @@
int closedir(FAR DIR *dirp)
{
int ret;
#ifdef CONFIG_FDSAN
uint64_t tag;
#endif

if (dirp == NULL)
{
set_errno(EBADF);
return -1;
}

#ifdef CONFIG_FDSAN
tag = android_fdsan_create_owner_tag(ANDROID_FDSAN_OWNER_TYPE_DIR,
(uintptr_t)dirp);
ret = android_fdsan_close_with_tag(dirp->fd, tag);
#else
ret = close(dirp->fd);
#endif

lib_free(dirp);
return ret;
}
12 changes: 12 additions & 0 deletions libs/libc/dirent/lib_opendir.c
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,11 @@
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>

#ifdef CONFIG_FDSAN
# include <android/fdsan.h>
#endif

#include <string.h>

#include "libc.h"
Expand Down Expand Up @@ -83,5 +88,12 @@ FAR DIR *opendir(FAR const char *path)
}

dir->fd = fd;

#ifdef CONFIG_FDSAN
android_fdsan_exchange_owner_tag(fd, 0,
android_fdsan_create_owner_tag(ANDROID_FDSAN_OWNER_TYPE_DIR,
(uintptr_t)dir));
#endif

return dir;
}
6 changes: 6 additions & 0 deletions libs/libc/misc/Kconfig
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,12 @@ config LIBC_ENVPATH
Use the contents of the common environment variable to locate executable
or library files. Default: n

config FDSAN
bool "Enable Fdsan"
default n
---help---
Enable the fdsan support

config LIBC_FTOK_VFS_PATH
string "Relative path to ftok storage"
default "/var/ftok"
Expand Down
6 changes: 6 additions & 0 deletions libs/libc/misc/Make.defs
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,12 @@ ifeq ($(CONFIG_LIBC_ENVPATH),y)
CSRCS += lib_envpath.c
endif

# Fdsan support

ifeq ($(CONFIG_FDSAN),y)
CSRCS += lib_fdsan.c
endif

# To ensure uname information is newest,
# add lib_utsname.o to phony target for force rebuild

Expand Down
Loading

0 comments on commit 90387a5

Please sign in to comment.