Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
OpenHarmony
Kernel Liteos A
提交
91696370
K
Kernel Liteos A
项目概览
OpenHarmony
/
Kernel Liteos A
1 年多 前同步成功
通知
461
Star
414
Fork
55
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
4
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
K
Kernel Liteos A
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
4
Issue
4
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
91696370
编写于
6月 17, 2021
作者:
O
openharmony_ci
提交者:
Gitee
6月 17, 2021
浏览文件
操作
浏览文件
下载
差异文件
!310 fatfs支持symlink与readlink功能
Merge pull request !310 from JING/link
上级
acb24924
e50cf0be
变更
1
显示空白变更内容
内联
并排
Showing
1 changed file
with
275 addition
and
257 deletion
+275
-257
fs/fat/os_adapt/fatfs.c
fs/fat/os_adapt/fatfs.c
+275
-257
未找到文件。
fs/fat/os_adapt/fatfs.c
浏览文件 @
91696370
...
@@ -231,117 +231,239 @@ static mode_t fatfs_get_mode(BYTE attribute, mode_t fs_mode)
...
@@ -231,117 +231,239 @@ static mode_t fatfs_get_mode(BYTE attribute, mode_t fs_mode)
fs_mode
&=
~
mask
;
fs_mode
&=
~
mask
;
if
(
attribute
&
AM_DIR
)
{
if
(
attribute
&
AM_DIR
)
{
fs_mode
|=
S_IFDIR
;
fs_mode
|=
S_IFDIR
;
}
else
if
(
attribute
&
AM_LNK
)
{
fs_mode
|=
S_IFLNK
;
}
else
{
}
else
{
fs_mode
|=
S_IFREG
;
fs_mode
|=
S_IFREG
;
}
}
return
fs_mode
;
return
fs_mode
;
}
}
int
fatfs_lookup
(
struct
Vnode
*
parent
,
const
char
*
path
,
int
len
,
struct
Vnode
**
vpp
)
static
enum
VnodeType
fatfstype_2_vnodetype
(
BYTE
type
)
{
switch
(
type
)
{
case
AM_ARC
:
return
VNODE_TYPE_REG
;
case
AM_DIR
:
return
VNODE_TYPE_DIR
;
case
AM_LNK
:
return
VNODE_TYPE_LNK
;
default:
return
VNODE_TYPE_UNKNOWN
;
}
}
static
FRESULT
init_cluster
(
DIR
*
dp_new
,
FATFS
*
fs
,
int
type
,
const
char
*
target
,
DWORD
*
clust
)
{
FRESULT
result
;
BYTE
*
dir
=
NULL
;
QWORD
sect
;
UINT
n
;
/* Allocate a new cluster */
*
clust
=
create_chain
(
&
(
dp_new
->
obj
),
0
);
if
(
*
clust
==
0
)
{
return
FR_NO_SPACE_LEFT
;
}
if
(
*
clust
==
1
||
*
clust
==
DISK_ERROR
)
{
return
FR_DISK_ERR
;
}
result
=
sync_window
(
fs
);
/* Flush FAT */
if
(
result
!=
FR_OK
)
{
remove_chain
(
&
(
dp_new
->
obj
),
*
clust
,
0
);
return
result
;
}
/* Initialize the new cluster */
#ifndef LOSCFG_FS_FAT_VIRTUAL_PARTITION
dir
=
fs
->
win
;
#else
dir
=
PARENTFS
(
fs
)
->
win
;
#endif
sect
=
clst2sect
(
fs
,
*
clust
);
mem_set
(
dir
,
0
,
SS
(
fs
));
if
(
type
==
AM_LNK
&&
target
)
{
/* Write target to symlink */
strcpy_s
((
char
*
)
dir
,
SS
(
fs
),
target
);
}
for
(
n
=
fs
->
csize
;
n
>
0
;
n
--
)
{
#ifndef LOSCFG_FS_FAT_VIRTUAL_PARTITION
fs
->
winsect
=
sect
++
;
fs
->
wflag
=
1
;
#else
PARENTFS
(
fs
)
->
winsect
=
sect
++
;
PARENTFS
(
fs
)
->
wflag
=
1
;
result
=
sync_window
(
fs
);
if
(
result
!=
FR_OK
)
{
remove_chain
(
&
(
dp_new
->
obj
),
*
clust
,
0
);
return
result
;
}
#endif
if
(
type
==
AM_LNK
)
{
/* No need to clean the rest sectors of the cluster for symlink */
break
;
}
}
return
FR_OK
;
}
static
int
fatfs_create_obj
(
struct
Vnode
*
parent
,
const
char
*
name
,
int
mode
,
struct
Vnode
**
vpp
,
BYTE
type
,
const
char
*
target
)
{
{
struct
Vnode
*
vp
=
NULL
;
struct
Vnode
*
vp
=
NULL
;
FATFS
*
fs
=
(
FATFS
*
)(
parent
->
originMount
->
data
);
FATFS
*
fs
=
(
FATFS
*
)
parent
->
originMount
->
data
;
DIR_FILE
*
dfp
;
DIR_FILE
*
dfp
=
(
DIR_FILE
*
)
parent
->
data
;
DIR
*
dp
=
NULL
;
FILINFO
*
finfo
=
&
(
dfp
->
fno
);
FILINFO
*
finfo
=
NULL
;
DIR_FILE
*
dfp_new
=
NULL
;
FILINFO
*
finfo_new
=
NULL
;
DIR
*
dp_new
=
NULL
;
QWORD
time
;
DWORD
hash
;
DWORD
hash
;
DWORD
clust
=
0
;
FRESULT
result
;
FRESULT
result
;
int
ret
;
int
ret
;
dfp
=
(
DIR_FILE
*
)
zalloc
(
sizeof
(
DIR_FILE
));
if
((
type
!=
AM_ARC
)
&&
(
type
!=
AM_DIR
)
&&
(
type
!=
AM_LNK
))
{
if
(
dfp
==
NULL
)
{
result
=
FR_INVALID_NAME
;
ret
=
ENOMEM
;
goto
ERROR_EXIT
;
}
dfp_new
=
(
DIR_FILE
*
)
zalloc
(
sizeof
(
DIR_FILE
));
if
(
dfp_new
==
NULL
)
{
result
=
FR_NOT_ENOUGH_CORE
;
goto
ERROR_EXIT
;
goto
ERROR_EXIT
;
}
}
ret
=
lock_fs
(
fs
);
ret
=
lock_fs
(
fs
);
if
(
ret
==
FALSE
)
{
if
(
ret
==
FALSE
)
{
/* lock failed */
re
t
=
EBUSY
;
re
sult
=
FR_TIMEOUT
;
goto
ERROR_FREE
;
goto
ERROR_FREE
;
}
}
finfo
=
&
(
dfp
->
fno
);
LOS_ListInit
(
&
finfo
->
fp_list
);
if
(
finfo
->
fattrib
&
AM_ARC
||
finfo
->
fattrib
&
AM_LNK
)
{
dp
=
&
(
dfp
->
f_dir
);
result
=
FR_NO_DIR
;
dp
->
obj
.
fs
=
fs
;
goto
ERROR_UNLOCK
;
dp
->
obj
.
sclust
=
((
DIR_FILE
*
)(
parent
->
data
))
->
fno
.
sclst
;
}
finfo_new
=
&
(
dfp_new
->
fno
);
LOS_ListInit
(
&
finfo_new
->
fp_list
);
dp_new
=
&
(
dfp_new
->
f_dir
);
dp_new
->
obj
.
fs
=
fs
;
dp_new
->
obj
.
sclust
=
finfo
->
sclst
;
DEF_NAMBUF
;
DEF_NAMBUF
;
INIT_NAMBUF
(
fs
);
INIT_NAMBUF
(
fs
);
result
=
create_name
(
dp
,
&
path
);
result
=
create_name
(
dp_new
,
&
name
);
if
(
result
!=
FR_OK
)
{
if
(
result
!=
FR_OK
)
{
ret
=
fatfs_2_vfs
(
result
);
goto
ERROR_UNLOCK
;
goto
ERROR_UNLOCK
;
}
}
result
=
dir_find
(
dp
);
result
=
dir_find
(
dp
_new
);
if
(
result
!
=
FR_OK
)
{
if
(
result
=
=
FR_OK
)
{
re
t
=
fatfs_2_vfs
(
result
)
;
re
sult
=
FR_EXIST
;
goto
ERROR_UNLOCK
;
goto
ERROR_UNLOCK
;
}
}
if
(
dp
->
fn
[
NSFLAG
]
&
NS_NONAME
)
{
if
(
type
==
AM_DIR
||
type
==
AM_LNK
)
{
result
=
FR_INVALID_NAME
;
result
=
init_cluster
(
dp_new
,
fs
,
type
,
target
,
&
clust
)
;
ret
=
fatfs_2_vfs
(
result
);
if
(
result
!=
FR_OK
)
{
goto
ERROR_UNLOCK
;
goto
ERROR_UNLOCK
;
}
}
}
get_fileinfo
(
dp
,
finfo
);
result
=
dir_register
(
dp_new
);
dp
->
obj
.
objsize
=
0
;
if
(
result
!=
FR_OK
)
{
goto
ERROR_REMOVE_CHAIN
;
}
/* Set the directory entry attribute */
if
(
time_status
==
SYSTEM_TIME_ENABLE
)
{
time
=
GET_FATTIME
();
}
else
{
time
=
0
;
}
st_dword
(
dp_new
->
dir
+
DIR_CrtTime
,
time
);
st_dword
(
dp_new
->
dir
+
DIR_ModTime
,
time
);
st_word
(
dp_new
->
dir
+
DIR_LstAccDate
,
time
>>
FTIME_DATE_OFFSET
);
dp_new
->
dir
[
DIR_Attr
]
=
type
;
if
(((
DWORD
)
mode
&
S_IWUSR
)
==
0
)
{
dp_new
->
dir
[
DIR_Attr
]
|=
AM_RDO
;
}
st_clust
(
fs
,
dp_new
->
dir
,
clust
);
if
(
type
==
AM_ARC
)
{
st_dword
(
dp_new
->
dir
+
DIR_FileSize
,
0
);
}
else
if
(
type
==
AM_LNK
)
{
st_dword
(
dp_new
->
dir
+
DIR_FileSize
,
strlen
(
target
));
}
#ifdef LOSCFG_FS_FAT_VIRTUAL_PARTITION
PARENTFS
(
fs
)
->
wflag
=
1
;
#else
fs
->
wflag
=
1
;
#endif
result
=
sync_fs
(
fs
);
if
(
result
!=
FR_OK
)
{
goto
ERROR_REMOVE_CHAIN
;
}
result
=
dir_read
(
dp_new
,
0
);
if
(
result
!=
FR_OK
)
{
goto
ERROR_REMOVE_CHAIN
;
}
dp_new
->
blk_ofs
=
dir_ofs
(
dp_new
);
get_fileinfo
(
dp_new
,
finfo_new
);
if
(
type
==
AM_ARC
)
{
dp_new
->
obj
.
objsize
=
0
;
}
else
if
(
type
==
AM_LNK
)
{
dp_new
->
obj
.
objsize
=
strlen
(
target
);
}
hash
=
fatfs_hash
(
dp
->
sect
,
dp
->
dptr
,
finfo
->
sclst
);
ret
=
VfsHashGet
(
parent
->
originMount
,
hash
,
&
vp
,
fatfs_hash_cmp
,
dfp
);
if
(
ret
!=
0
)
{
ret
=
VnodeAlloc
(
&
fatfs_vops
,
&
vp
);
ret
=
VnodeAlloc
(
&
fatfs_vops
,
&
vp
);
if
(
ret
!=
0
)
{
if
(
ret
!=
0
)
{
ret
=
ENOMEM
;
result
=
FR_NOT_ENOUGH_CORE
;
result
=
FR_NOT_ENOUGH_CORE
;
goto
ERROR_UNLOCK
;
goto
ERROR_REMOVE_CHAIN
;
}
}
vp
->
parent
=
parent
;
vp
->
parent
=
parent
;
vp
->
fop
=
&
fatfs_fops
;
vp
->
fop
=
&
fatfs_fops
;
vp
->
data
=
dfp
;
vp
->
data
=
dfp_new
;
vp
->
originMount
=
parent
->
originMount
;
vp
->
originMount
=
parent
->
originMount
;
vp
->
uid
=
fs
->
fs_uid
;
vp
->
uid
=
fs
->
fs_uid
;
vp
->
gid
=
fs
->
fs_gid
;
vp
->
gid
=
fs
->
fs_gid
;
vp
->
mode
=
fatfs_get_mode
(
finfo
->
fattrib
,
fs
->
fs_mode
);
vp
->
mode
=
fatfs_get_mode
(
finfo_new
->
fattrib
,
fs
->
fs_mode
);
if
(
finfo
->
fattrib
&
AM_DIR
)
{
vp
->
type
=
fatfstype_2_vnodetype
(
type
);
vp
->
type
=
VNODE_TYPE_DIR
;
}
else
{
vp
->
type
=
VNODE_TYPE_REG
;
}
hash
=
fatfs_hash
(
dp_new
->
sect
,
dp_new
->
dptr
,
finfo_new
->
sclst
);
ret
=
VfsHashInsert
(
vp
,
hash
);
ret
=
VfsHashInsert
(
vp
,
hash
);
if
(
ret
!=
0
)
{
if
(
ret
!=
0
)
{
result
=
FR_INVALID_PARAMETER
;
result
=
FR_NOT_ENOUGH_CORE
;
goto
ERROR_UNLOCK
;
goto
ERROR_REMOVE_CHAIN
;
}
}
else
{
vp
->
parent
=
parent
;
free
(
dfp
);
/* hash hit dfp is no needed */
}
}
*
vpp
=
vp
;
unlock_fs
(
fs
,
FR_OK
);
unlock_fs
(
fs
,
FR_OK
);
FREE_NAMBUF
();
FREE_NAMBUF
();
*
vpp
=
vp
;
return
fatfs_sync
(
parent
->
originMount
->
mountFlags
,
fs
);
return
0
;
ERROR_REMOVE_CHAIN:
remove_chain
(
&
(
dp_new
->
obj
),
clust
,
0
);
ERROR_UNLOCK:
ERROR_UNLOCK:
unlock_fs
(
fs
,
result
);
unlock_fs
(
fs
,
result
);
FREE_NAMBUF
();
FREE_NAMBUF
();
ERROR_FREE:
ERROR_FREE:
free
(
dfp
);
free
(
dfp
_new
);
ERROR_EXIT:
ERROR_EXIT:
return
-
ret
;
return
-
fatfs_2_vfs
(
result
)
;
}
}
int
fatfs_
create
(
struct
Vnode
*
parent
,
const
char
*
name
,
int
mode
,
struct
Vnode
**
vpp
)
int
fatfs_
lookup
(
struct
Vnode
*
parent
,
const
char
*
path
,
int
len
,
struct
Vnode
**
vpp
)
{
{
struct
Vnode
*
vp
=
NULL
;
struct
Vnode
*
vp
=
NULL
;
FATFS
*
fs
=
(
FATFS
*
)
parent
->
originMount
->
data
;
FATFS
*
fs
=
(
FATFS
*
)
(
parent
->
originMount
->
data
)
;
DIR_FILE
*
dfp
;
DIR_FILE
*
dfp
;
DIR
*
dp
=
NULL
;
DIR
*
dp
=
NULL
;
FILINFO
*
finfo
=
NULL
;
FILINFO
*
finfo
=
NULL
;
QWORD
time
;
DWORD
hash
;
DWORD
hash
;
FRESULT
result
;
FRESULT
result
;
int
ret
;
int
ret
;
...
@@ -351,12 +473,12 @@ int fatfs_create(struct Vnode *parent, const char *name, int mode, struct Vnode
...
@@ -351,12 +473,12 @@ int fatfs_create(struct Vnode *parent, const char *name, int mode, struct Vnode
ret
=
ENOMEM
;
ret
=
ENOMEM
;
goto
ERROR_EXIT
;
goto
ERROR_EXIT
;
}
}
ret
=
lock_fs
(
fs
);
ret
=
lock_fs
(
fs
);
if
(
ret
==
FALSE
)
{
/* lock failed */
if
(
ret
==
FALSE
)
{
ret
=
EBUSY
;
ret
=
EBUSY
;
goto
ERROR_FREE
;
goto
ERROR_FREE
;
}
}
finfo
=
&
(
dfp
->
fno
);
finfo
=
&
(
dfp
->
fno
);
LOS_ListInit
(
&
finfo
->
fp_list
);
LOS_ListInit
(
&
finfo
->
fp_list
);
dp
=
&
(
dfp
->
f_dir
);
dp
=
&
(
dfp
->
f_dir
);
...
@@ -365,62 +487,36 @@ int fatfs_create(struct Vnode *parent, const char *name, int mode, struct Vnode
...
@@ -365,62 +487,36 @@ int fatfs_create(struct Vnode *parent, const char *name, int mode, struct Vnode
DEF_NAMBUF
;
DEF_NAMBUF
;
INIT_NAMBUF
(
fs
);
INIT_NAMBUF
(
fs
);
result
=
create_name
(
dp
,
&
name
);
result
=
create_name
(
dp
,
&
path
);
if
(
result
!=
FR_OK
)
{
if
(
result
!=
FR_OK
)
{
ret
=
fatfs_2_vfs
(
result
);
ret
=
fatfs_2_vfs
(
result
);
goto
ERROR_UNLOCK
;
goto
ERROR_UNLOCK
;
}
}
result
=
dir_find
(
dp
);
result
=
dir_find
(
dp
);
if
(
result
==
FR_OK
)
{
ret
=
EEXIST
;
goto
ERROR_UNLOCK
;
}
result
=
dir_register
(
dp
);
if
(
result
!=
FR_OK
)
{
if
(
result
!=
FR_OK
)
{
ret
=
fatfs_2_vfs
(
result
);
ret
=
fatfs_2_vfs
(
result
);
goto
ERROR_UNLOCK
;
goto
ERROR_UNLOCK
;
}
}
/* Set the directory entry attribute */
if
(
time_status
==
SYSTEM_TIME_ENABLE
)
{
time
=
GET_FATTIME
();
}
else
{
time
=
0
;
}
st_dword
(
dp
->
dir
+
DIR_CrtTime
,
time
);
st_dword
(
dp
->
dir
+
DIR_ModTime
,
time
);
st_word
(
dp
->
dir
+
DIR_LstAccDate
,
time
>>
FTIME_DATE_OFFSET
);
dp
->
dir
[
DIR_Attr
]
=
AM_ARC
;
if
(((
DWORD
)
mode
&
S_IWUSR
)
==
0
)
{
dp
->
dir
[
DIR_Attr
]
|=
AM_RDO
;
}
st_clust
(
fs
,
dp
->
dir
,
0
);
st_dword
(
dp
->
dir
+
DIR_FileSize
,
0
);
#ifdef LOSCFG_FS_FAT_VIRTUAL_PARTITION
if
(
dp
->
fn
[
NSFLAG
]
&
NS_NONAME
)
{
PARENTFS
(
fs
)
->
wflag
=
1
;
result
=
FR_INVALID_NAME
;
#else
fs
->
wflag
=
1
;
#endif
result
=
sync_fs
(
fs
);
if
(
result
!=
FR_OK
)
{
ret
=
fatfs_2_vfs
(
result
);
goto
ERROR_UNLOCK
;
}
result
=
dir_read
(
dp
,
0
);
if
(
result
!=
FR_OK
)
{
ret
=
fatfs_2_vfs
(
result
);
ret
=
fatfs_2_vfs
(
result
);
goto
ERROR_UNLOCK
;
goto
ERROR_UNLOCK
;
}
}
dp
->
blk_ofs
=
dir_ofs
(
dp
);
get_fileinfo
(
dp
,
finfo
);
get_fileinfo
(
dp
,
finfo
);
dp
->
obj
.
objsize
=
0
;
dp
->
obj
.
objsize
=
0
;
hash
=
fatfs_hash
(
dp
->
sect
,
dp
->
dptr
,
finfo
->
sclst
);
ret
=
VfsHashGet
(
parent
->
originMount
,
hash
,
&
vp
,
fatfs_hash_cmp
,
dfp
);
if
(
ret
!=
0
)
{
ret
=
VnodeAlloc
(
&
fatfs_vops
,
&
vp
);
ret
=
VnodeAlloc
(
&
fatfs_vops
,
&
vp
);
if
(
ret
!=
0
)
{
if
(
ret
!=
0
)
{
ret
=
ENOMEM
;
ret
=
ENOMEM
;
result
=
FR_NOT_ENOUGH_CORE
;
goto
ERROR_UNLOCK
;
goto
ERROR_UNLOCK
;
}
}
vp
->
parent
=
parent
;
vp
->
parent
=
parent
;
vp
->
fop
=
&
fatfs_fops
;
vp
->
fop
=
&
fatfs_fops
;
vp
->
data
=
dfp
;
vp
->
data
=
dfp
;
...
@@ -428,19 +524,26 @@ int fatfs_create(struct Vnode *parent, const char *name, int mode, struct Vnode
...
@@ -428,19 +524,26 @@ int fatfs_create(struct Vnode *parent, const char *name, int mode, struct Vnode
vp
->
uid
=
fs
->
fs_uid
;
vp
->
uid
=
fs
->
fs_uid
;
vp
->
gid
=
fs
->
fs_gid
;
vp
->
gid
=
fs
->
fs_gid
;
vp
->
mode
=
fatfs_get_mode
(
finfo
->
fattrib
,
fs
->
fs_mode
);
vp
->
mode
=
fatfs_get_mode
(
finfo
->
fattrib
,
fs
->
fs_mode
);
if
(
finfo
->
fattrib
&
AM_DIR
)
{
vp
->
type
=
VNODE_TYPE_DIR
;
}
else
{
vp
->
type
=
VNODE_TYPE_REG
;
vp
->
type
=
VNODE_TYPE_REG
;
}
hash
=
fatfs_hash
(
dp
->
sect
,
dp
->
dptr
,
finfo
->
sclst
);
ret
=
VfsHashInsert
(
vp
,
hash
);
ret
=
VfsHashInsert
(
vp
,
hash
);
if
(
ret
!=
0
)
{
if
(
ret
!=
0
)
{
ret
=
EINVAL
;
result
=
FR_INVALID_PARAMETER
;
goto
ERROR_UNLOCK
;
goto
ERROR_UNLOCK
;
}
}
*
vpp
=
vp
;
}
else
{
vp
->
parent
=
parent
;
free
(
dfp
);
/* hash hit dfp is no needed */
}
unlock_fs
(
fs
,
result
);
unlock_fs
(
fs
,
FR_OK
);
FREE_NAMBUF
();
FREE_NAMBUF
();
return
fatfs_sync
(
parent
->
originMount
->
mountFlags
,
fs
);
*
vpp
=
vp
;
return
0
;
ERROR_UNLOCK:
ERROR_UNLOCK:
unlock_fs
(
fs
,
result
);
unlock_fs
(
fs
,
result
);
...
@@ -451,6 +554,11 @@ ERROR_EXIT:
...
@@ -451,6 +554,11 @@ ERROR_EXIT:
return
-
ret
;
return
-
ret
;
}
}
int
fatfs_create
(
struct
Vnode
*
parent
,
const
char
*
name
,
int
mode
,
struct
Vnode
**
vpp
)
{
return
fatfs_create_obj
(
parent
,
name
,
mode
,
vpp
,
AM_ARC
,
NULL
);
}
int
fatfs_open
(
struct
file
*
filep
)
int
fatfs_open
(
struct
file
*
filep
)
{
{
struct
Vnode
*
vp
=
filep
->
f_vnode
;
struct
Vnode
*
vp
=
filep
->
f_vnode
;
...
@@ -1690,150 +1798,7 @@ int fatfs_mkfs (struct Vnode *device, int sectors, int option)
...
@@ -1690,150 +1798,7 @@ int fatfs_mkfs (struct Vnode *device, int sectors, int option)
int
fatfs_mkdir
(
struct
Vnode
*
parent
,
const
char
*
name
,
mode_t
mode
,
struct
Vnode
**
vpp
)
int
fatfs_mkdir
(
struct
Vnode
*
parent
,
const
char
*
name
,
mode_t
mode
,
struct
Vnode
**
vpp
)
{
{
struct
Vnode
*
vp
=
NULL
;
return
fatfs_create_obj
(
parent
,
name
,
mode
,
vpp
,
AM_DIR
,
NULL
);
FATFS
*
fs
=
(
FATFS
*
)
parent
->
originMount
->
data
;
DIR_FILE
*
dfp
=
(
DIR_FILE
*
)
parent
->
data
;
FILINFO
*
finfo
=
&
(
dfp
->
fno
);
DIR_FILE
*
dfp_new
=
NULL
;
QWORD
sect
;
DWORD
clust
;
BYTE
*
dir
=
NULL
;
DWORD
hash
;
FRESULT
result
=
FR_OK
;
int
ret
;
UINT
n
;
ret
=
lock_fs
(
fs
);
if
(
ret
==
FALSE
)
{
result
=
FR_TIMEOUT
;
goto
ERROR_OUT
;
}
if
(
finfo
->
fattrib
&
AM_ARC
)
{
result
=
FR_NO_DIR
;
goto
ERROR_UNLOCK
;
}
DEF_NAMBUF
;
INIT_NAMBUF
(
fs
);
dfp_new
=
(
DIR_FILE
*
)
zalloc
(
sizeof
(
DIR_FILE
));
if
(
dfp_new
==
NULL
)
{
result
=
FR_NOT_ENOUGH_CORE
;
goto
ERROR_UNLOCK
;
}
LOS_ListInit
(
&
(
dfp_new
->
fno
.
fp_list
));
dfp_new
->
f_dir
.
obj
.
sclust
=
finfo
->
sclst
;
dfp_new
->
f_dir
.
obj
.
fs
=
fs
;
result
=
create_name
(
&
(
dfp_new
->
f_dir
),
&
name
);
if
(
result
!=
FR_OK
)
{
goto
ERROR_FREE
;
}
result
=
dir_find
(
&
(
dfp_new
->
f_dir
));
if
(
result
==
FR_OK
)
{
result
=
FR_EXIST
;
goto
ERROR_FREE
;
}
/* Allocate new chain for directory */
clust
=
create_chain
(
&
(
dfp_new
->
f_dir
.
obj
),
0
);
if
(
clust
==
0
)
{
result
=
FR_NO_SPACE_LEFT
;
goto
ERROR_FREE
;
}
if
(
clust
==
1
||
clust
==
DISK_ERROR
)
{
result
=
FR_DISK_ERR
;
goto
ERROR_FREE
;
}
result
=
sync_window
(
fs
);
/* Flush FAT */
if
(
result
!=
FR_OK
)
{
goto
ERROR_REMOVE_CHAIN
;
}
/* Initialize the new directory */
#ifndef LOSCFG_FS_FAT_VIRTUAL_PARTITION
dir
=
fs
->
win
;
#else
dir
=
PARENTFS
(
fs
)
->
win
;
#endif
sect
=
clst2sect
(
fs
,
clust
);
mem_set
(
dir
,
0
,
SS
(
fs
));
for
(
n
=
fs
->
csize
;
n
>
0
;
n
--
)
{
/* Write zero to directory */
#ifndef LOSCFG_FS_FAT_VIRTUAL_PARTITION
fs
->
winsect
=
sect
++
;
fs
->
wflag
=
1
;
#else
PARENTFS
(
fs
)
->
winsect
=
sect
++
;
PARENTFS
(
fs
)
->
wflag
=
1
;
result
=
sync_window
(
fs
);
if
(
result
!=
FR_OK
)
break
;
#endif
}
if
(
result
!=
FR_OK
)
{
goto
ERROR_REMOVE_CHAIN
;
}
result
=
dir_register
(
&
(
dfp_new
->
f_dir
));
if
(
result
!=
FR_OK
)
{
goto
ERROR_REMOVE_CHAIN
;
}
dir
=
dfp_new
->
f_dir
.
dir
;
st_dword
(
dir
+
DIR_ModTime
,
0
);
/* Set the time */
st_clust
(
fs
,
dir
,
clust
);
/* Set the start cluster */
dir
[
DIR_Attr
]
=
AM_DIR
;
/* Set the attrib */
if
((
mode
&
S_IWUSR
)
==
0
)
{
dir
[
DIR_Attr
]
|=
AM_RDO
;
}
#ifndef LOSCFG_FS_FAT_VIRTUAL_PARTITION
fs
->
wflag
=
1
;
#else
PARENTFS
(
fs
)
->
wflag
=
1
;
#endif
result
=
sync_fs
(
fs
);
if
(
result
!=
FR_OK
)
{
goto
ERROR_REMOVE_CHAIN
;
}
/* Set the FILINFO struct */
result
=
dir_read
(
&
(
dfp_new
->
f_dir
),
0
);
if
(
result
!=
FR_OK
)
{
goto
ERROR_REMOVE_CHAIN
;
}
dfp_new
->
f_dir
.
blk_ofs
=
dir_ofs
(
&
(
dfp_new
->
f_dir
));
get_fileinfo
(
&
(
dfp_new
->
f_dir
),
&
(
dfp_new
->
fno
));
ret
=
VnodeAlloc
(
&
fatfs_vops
,
&
vp
);
if
(
ret
!=
0
)
{
result
=
FR_NOT_ENOUGH_CORE
;
goto
ERROR_REMOVE_CHAIN
;
}
vp
->
parent
=
parent
;
vp
->
fop
=
&
fatfs_fops
;
vp
->
data
=
dfp_new
;
vp
->
originMount
=
parent
->
originMount
;
vp
->
uid
=
fs
->
fs_uid
;
vp
->
gid
=
fs
->
fs_gid
;
vp
->
mode
=
fatfs_get_mode
(
dfp_new
->
fno
.
fattrib
,
fs
->
fs_mode
);
vp
->
type
=
VNODE_TYPE_DIR
;
hash
=
fatfs_hash
(
dfp_new
->
f_dir
.
sect
,
dfp_new
->
f_dir
.
dptr
,
dfp_new
->
fno
.
sclst
);
ret
=
VfsHashInsert
(
vp
,
hash
);
if
(
ret
!=
0
)
{
result
=
FR_NOT_ENOUGH_CORE
;
goto
ERROR_REMOVE_CHAIN
;
}
unlock_fs
(
fs
,
FR_OK
);
FREE_NAMBUF
();
*
vpp
=
vp
;
return
fatfs_sync
(
vp
->
originMount
->
mountFlags
,
fs
);
ERROR_REMOVE_CHAIN:
remove_chain
(
&
(
dfp_new
->
f_dir
.
obj
),
clust
,
0
);
ERROR_FREE:
free
(
dfp_new
);
ERROR_UNLOCK:
unlock_fs
(
fs
,
result
);
FREE_NAMBUF
();
ERROR_OUT:
return
-
fatfs_2_vfs
(
result
);
}
}
int
fatfs_rmdir
(
struct
Vnode
*
parent
,
struct
Vnode
*
vp
,
const
char
*
name
)
int
fatfs_rmdir
(
struct
Vnode
*
parent
,
struct
Vnode
*
vp
,
const
char
*
name
)
...
@@ -2064,6 +2029,57 @@ ERROR_WITH_DIR:
...
@@ -2064,6 +2029,57 @@ ERROR_WITH_DIR:
return
-
fatfs_2_vfs
(
result
);
return
-
fatfs_2_vfs
(
result
);
}
}
int
fatfs_symlink
(
struct
Vnode
*
parentVnode
,
struct
Vnode
**
newVnode
,
const
char
*
path
,
const
char
*
target
)
{
return
fatfs_create_obj
(
parentVnode
,
path
,
0
,
newVnode
,
AM_LNK
,
target
);
}
ssize_t
fatfs_readlink
(
struct
Vnode
*
vnode
,
char
*
buffer
,
size_t
bufLen
)
{
int
ret
;
FRESULT
res
=
FR_OK
;
DWORD
clust
;
QWORD
sect
;
DIR_FILE
*
dfp
=
(
DIR_FILE
*
)(
vnode
->
data
);
DIR
*
dp
=
&
(
dfp
->
f_dir
);
FATFS
*
fs
=
dp
->
obj
.
fs
;
FILINFO
*
finfo
=
&
(
dfp
->
fno
);
size_t
targetLen
=
finfo
->
fsize
;
size_t
cnt
;
ret
=
lock_fs
(
fs
);
if
(
ret
==
FALSE
)
{
return
-
EBUSY
;
}
clust
=
finfo
->
sclst
;
sect
=
clst2sect
(
fs
,
clust
);
/* Get current sector */
if
(
sect
==
0
)
{
res
=
FR_DISK_ERR
;
goto
ERROUT
;
}
if
(
move_window
(
fs
,
sect
)
!=
FR_OK
)
{
res
=
FR_DISK_ERR
;
goto
ERROUT
;
}
cnt
=
(
bufLen
-
1
)
<
targetLen
?
(
bufLen
-
1
)
:
targetLen
;
ret
=
LOS_CopyFromKernel
(
buffer
,
bufLen
,
fs
->
win
,
cnt
);
if
(
ret
!=
EOK
)
{
res
=
FR_INVALID_PARAMETER
;
goto
ERROUT
;
}
buffer
[
cnt
]
=
'\0'
;
unlock_fs
(
fs
,
FR_OK
);
return
cnt
;
ERROUT:
unlock_fs
(
fs
,
res
);
return
-
fatfs_2_vfs
(
res
);
}
struct
VnodeOps
fatfs_vops
=
{
struct
VnodeOps
fatfs_vops
=
{
/* file ops */
/* file ops */
.
Getattr
=
fatfs_stat
,
.
Getattr
=
fatfs_stat
,
...
@@ -2083,6 +2099,8 @@ struct VnodeOps fatfs_vops = {
...
@@ -2083,6 +2099,8 @@ struct VnodeOps fatfs_vops = {
.
Mkdir
=
fatfs_mkdir
,
.
Mkdir
=
fatfs_mkdir
,
.
Rmdir
=
fatfs_rmdir
,
.
Rmdir
=
fatfs_rmdir
,
.
Fscheck
=
fatfs_fscheck
,
.
Fscheck
=
fatfs_fscheck
,
.
Symlink
=
fatfs_symlink
,
.
Readlink
=
fatfs_readlink
,
};
};
struct
MountOps
fatfs_mops
=
{
struct
MountOps
fatfs_mops
=
{
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录