Add computing offloading code
1. Add computing offloading code 2. Add script.md 3. Add virsh_demo.xml Change-Id: Id9ef883e2f0eb727eb5448b9d1c47767f46b1021 Signed-off-by: Yikun Jiang <yikunkero@gmail.com>
This commit is contained in:
parent
0b073c658d
commit
a68570b5d9
@ -22,17 +22,25 @@ Computing Offload and Integration Subgroup (算力卸载与集成子工作组)
|
||||
|
||||
1. 无感卸载代码
|
||||
|
||||
代码链接:[链接](./qtfs/)
|
||||
|
||||
将libvirt从host侧卸载至DPU侧,host与DPU之间通过一个插件进行交互;在OIF开源libvirt实现,以及OpenEuler和DPU之间的插件
|
||||
|
||||
2. 无感卸载方案/代码说明文档
|
||||
|
||||
文档链接:[链接](./qtfs/doc)
|
||||
|
||||
对上述技术方案及代码实现的说明文档
|
||||
|
||||
3. openEuler/大禹/九州云demo联创演示及视频
|
||||
|
||||
演示脚本: [链接](./script.md)
|
||||
|
||||
基于无感卸载方案、使用大禹DPU卡、九州云OpenStack平台完成演示
|
||||
|
||||
4. DPU工作组背景介绍
|
||||
|
||||
文档链接:[链接](./README.md)
|
||||
|
||||
描述子项目背景、project facts、scope、与CFN关系、Release plan等
|
||||
|
||||
|
40
qtfs/CMakeLists.txt
Normal file
40
qtfs/CMakeLists.txt
Normal file
@ -0,0 +1,40 @@
|
||||
cmake_minimum_required(VERSION 3.0.0)
|
||||
|
||||
project(qtfs)
|
||||
|
||||
set(CMAKE_C_FLAGS "-g -O2 -fstack-protector-strong -fPIE -pie -fPIC -D_FORTIFY_SOURCE=2 -s -Wl,-z,now -Wl,-z,noexecstack")
|
||||
|
||||
# Build rexec and rexec_server
|
||||
add_executable(rexec rexec/rexec.c rexec/rexec_sock.c)
|
||||
add_executable(rexec_server rexec/rexec_server.c rexec/rexec_sock.c rexec/rexec_shim.c)
|
||||
target_include_directories(rexec_server PRIVATE /usr/include/glib-2.0 /usr/lib64/glib-2.0/include)
|
||||
target_link_libraries(rexec PRIVATE json-c pthread)
|
||||
target_link_libraries(rexec_server PRIVATE json-c glib-2.0)
|
||||
|
||||
# Build udsproxyd and libudsproxy.so
|
||||
add_executable(udsproxyd ipc/uds_event.c ipc/uds_main.c)
|
||||
add_library(udsproxy SHARED ipc/uds_connector.c)
|
||||
target_include_directories(udsproxyd PRIVATE include/ /usr/include/glib-2.0 /usr/lib64/glib-2.0/include)
|
||||
target_link_libraries(udsproxyd PRIVATE pthread glib-2.0)
|
||||
|
||||
# Build engine
|
||||
add_executable(engine ipc/uds_main.c ipc/uds_event.c qtfs_common/user_engine.c)
|
||||
target_include_directories(engine PRIVATE include/ ./ ipc/ /usr/include/glib-2.0 /usr/lib64/glib-2.0/include)
|
||||
target_link_libraries(engine PRIVATE glib-2.0 pthread)
|
||||
target_compile_options(engine PRIVATE "-DQTFS_SERVER")
|
||||
|
||||
if (DEFINED UDS_TEST_MODE OR DEFINED QTFS_TEST_MODE)
|
||||
target_compile_options(engine PRIVATE "-DUDS_TEST_MODE")
|
||||
target_compile_options(udsproxyd PRIVATE "-DUDS_TEST_MODE")
|
||||
message(WARNING "Important risk warning: the test mode is turned on, and qtfs will expose the network port, \
|
||||
which will bring security risks and is only for testing! If you do not understand the risks,\
|
||||
please don't use or compile again without test mode macro!")
|
||||
endif ()
|
||||
|
||||
# Build Kernel Module
|
||||
set(QTFS_BASE_DIR ${CMAKE_CURRENT_SOURCE_DIR})
|
||||
add_subdirectory(qtfs_server qtfs_server-km)
|
||||
add_subdirectory(qtfs qtfs-km)
|
||||
|
||||
|
||||
set(ignoreMe "${QTFS_TEST_MODE}${UDS_TEST_MODE}")
|
127
qtfs/LICENSE
Normal file
127
qtfs/LICENSE
Normal file
@ -0,0 +1,127 @@
|
||||
木兰宽松许可证, 第2版
|
||||
|
||||
木兰宽松许可证, 第2版
|
||||
2020年1月 http://license.coscl.org.cn/MulanPSL2
|
||||
|
||||
|
||||
您对“软件”的复制、使用、修改及分发受木兰宽松许可证,第2版(“本许可证”)的如下条款的约束:
|
||||
|
||||
0. 定义
|
||||
|
||||
“软件”是指由“贡献”构成的许可在“本许可证”下的程序和相关文档的集合。
|
||||
|
||||
“贡献”是指由任一“贡献者”许可在“本许可证”下的受版权法保护的作品。
|
||||
|
||||
“贡献者”是指将受版权法保护的作品许可在“本许可证”下的自然人或“法人实体”。
|
||||
|
||||
“法人实体”是指提交贡献的机构及其“关联实体”。
|
||||
|
||||
“关联实体”是指,对“本许可证”下的行为方而言,控制、受控制或与其共同受控制的机构,此处的控制是指有受控方或共同受控方至少50%直接或间接的投票权、资金或其他有价证券。
|
||||
|
||||
1. 授予版权许可
|
||||
|
||||
每个“贡献者”根据“本许可证”授予您永久性的、全球性的、免费的、非独占的、不可撤销的版权许可,您可以复制、使用、修改、分发其“贡献”,不论修改与否。
|
||||
|
||||
2. 授予专利许可
|
||||
|
||||
每个“贡献者”根据“本许可证”授予您永久性的、全球性的、免费的、非独占的、不可撤销的(根据本条规定撤销除外)专利许可,供您制造、委托制造、使用、许诺销售、销售、进口其“贡献”或以其他方式转移其“贡献”。前述专利许可仅限于“贡献者”现在或将来拥有或控制的其“贡献”本身或其“贡献”与许可“贡献”时的“软件”结合而将必然会侵犯的专利权利要求,不包括对“贡献”的修改或包含“贡献”的其他结合。如果您或您的“关联实体”直接或间接地,就“软件”或其中的“贡献”对任何人发起专利侵权诉讼(包括反诉或交叉诉讼)或其他专利维权行动,指控其侵犯专利权,则“本许可证”授予您对“软件”的专利许可自您提起诉讼或发起维权行动之日终止。
|
||||
|
||||
3. 无商标许可
|
||||
|
||||
“本许可证”不提供对“贡献者”的商品名称、商标、服务标志或产品名称的商标许可,但您为满足第4条规定的声明义务而必须使用除外。
|
||||
|
||||
4. 分发限制
|
||||
|
||||
您可以在任何媒介中将“软件”以源程序形式或可执行形式重新分发,不论修改与否,但您必须向接收者提供“本许可证”的副本,并保留“软件”中的版权、商标、专利及免责声明。
|
||||
|
||||
5. 免责声明与责任限制
|
||||
|
||||
“软件”及其中的“贡献”在提供时不带任何明示或默示的担保。在任何情况下,“贡献者”或版权所有者不对任何人因使用“软件”或其中的“贡献”而引发的任何直接或间接损失承担责任,不论因何种原因导致或者基于何种法律理论,即使其曾被建议有此种损失的可能性。
|
||||
|
||||
6. 语言
|
||||
“本许可证”以中英文双语表述,中英文版本具有同等法律效力。如果中英文版本存在任何冲突不一致,以中文版为准。
|
||||
|
||||
条款结束
|
||||
|
||||
如何将木兰宽松许可证,第2版,应用到您的软件
|
||||
|
||||
如果您希望将木兰宽松许可证,第2版,应用到您的新软件,为了方便接收者查阅,建议您完成如下三步:
|
||||
|
||||
1, 请您补充如下声明中的空白,包括软件名、软件的首次发表年份以及您作为版权人的名字;
|
||||
|
||||
2, 请您在软件包的一级目录下创建以“LICENSE”为名的文件,将整个许可证文本放入该文件中;
|
||||
|
||||
3, 请将如下声明文本放入每个源文件的头部注释中。
|
||||
|
||||
Copyright (c) [Year] [name of copyright holder]
|
||||
[Software Name] is licensed under Mulan PSL v2.
|
||||
You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
You may obtain a copy of Mulan PSL v2 at:
|
||||
http://license.coscl.org.cn/MulanPSL2
|
||||
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
||||
See the Mulan PSL v2 for more details.
|
||||
|
||||
|
||||
Mulan Permissive Software License,Version 2
|
||||
|
||||
Mulan Permissive Software License,Version 2 (Mulan PSL v2)
|
||||
January 2020 http://license.coscl.org.cn/MulanPSL2
|
||||
|
||||
Your reproduction, use, modification and distribution of the Software shall be subject to Mulan PSL v2 (this License) with the following terms and conditions:
|
||||
|
||||
0. Definition
|
||||
|
||||
Software means the program and related documents which are licensed under this License and comprise all Contribution(s).
|
||||
|
||||
Contribution means the copyrightable work licensed by a particular Contributor under this License.
|
||||
|
||||
Contributor means the Individual or Legal Entity who licenses its copyrightable work under this License.
|
||||
|
||||
Legal Entity means the entity making a Contribution and all its Affiliates.
|
||||
|
||||
Affiliates means entities that control, are controlled by, or are under common control with the acting entity under this License, ‘control’ means direct or indirect ownership of at least fifty percent (50%) of the voting power, capital or other securities of controlled or commonly controlled entity.
|
||||
|
||||
1. Grant of Copyright License
|
||||
|
||||
Subject to the terms and conditions of this License, each Contributor hereby grants to you a perpetual, worldwide, royalty-free, non-exclusive, irrevocable copyright license to reproduce, use, modify, or distribute its Contribution, with modification or not.
|
||||
|
||||
2. Grant of Patent License
|
||||
|
||||
Subject to the terms and conditions of this License, each Contributor hereby grants to you a perpetual, worldwide, royalty-free, non-exclusive, irrevocable (except for revocation under this Section) patent license to make, have made, use, offer for sale, sell, import or otherwise transfer its Contribution, where such patent license is only limited to the patent claims owned or controlled by such Contributor now or in future which will be necessarily infringed by its Contribution alone, or by combination of the Contribution with the Software to which the Contribution was contributed. The patent license shall not apply to any modification of the Contribution, and any other combination which includes the Contribution. If you or your Affiliates directly or indirectly institute patent litigation (including a cross claim or counterclaim in a litigation) or other patent enforcement activities against any individual or entity by alleging that the Software or any Contribution in it infringes patents, then any patent license granted to you under this License for the Software shall terminate as of the date such litigation or activity is filed or taken.
|
||||
|
||||
3. No Trademark License
|
||||
|
||||
No trademark license is granted to use the trade names, trademarks, service marks, or product names of Contributor, except as required to fulfill notice requirements in Section 4.
|
||||
|
||||
4. Distribution Restriction
|
||||
|
||||
You may distribute the Software in any medium with or without modification, whether in source or executable forms, provided that you provide recipients with a copy of this License and retain copyright, patent, trademark and disclaimer statements in the Software.
|
||||
|
||||
5. Disclaimer of Warranty and Limitation of Liability
|
||||
|
||||
THE SOFTWARE AND CONTRIBUTION IN IT ARE PROVIDED WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED. IN NO EVENT SHALL ANY CONTRIBUTOR OR COPYRIGHT HOLDER BE LIABLE TO YOU FOR ANY DAMAGES, INCLUDING, BUT NOT LIMITED TO ANY DIRECT, OR INDIRECT, SPECIAL OR CONSEQUENTIAL DAMAGES ARISING FROM YOUR USE OR INABILITY TO USE THE SOFTWARE OR THE CONTRIBUTION IN IT, NO MATTER HOW IT’S CAUSED OR BASED ON WHICH LEGAL THEORY, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
|
||||
|
||||
6. Language
|
||||
|
||||
THIS LICENSE IS WRITTEN IN BOTH CHINESE AND ENGLISH, AND THE CHINESE VERSION AND ENGLISH VERSION SHALL HAVE THE SAME LEGAL EFFECT. IN THE CASE OF DIVERGENCE BETWEEN THE CHINESE AND ENGLISH VERSIONS, THE CHINESE VERSION SHALL PREVAIL.
|
||||
|
||||
END OF THE TERMS AND CONDITIONS
|
||||
|
||||
How to Apply the Mulan Permissive Software License,Version 2 (Mulan PSL v2) to Your Software
|
||||
|
||||
To apply the Mulan PSL v2 to your work, for easy identification by recipients, you are suggested to complete following three steps:
|
||||
|
||||
i Fill in the blanks in following statement, including insert your software name, the year of the first publication of your software, and your name identified as the copyright owner;
|
||||
|
||||
ii Create a file named “LICENSE” which contains the whole context of this License in the first directory of your software package;
|
||||
|
||||
iii Attach the statement to the appropriate annotated syntax at the beginning of each source file.
|
||||
|
||||
|
||||
Copyright (c) [Year] [name of copyright holder]
|
||||
[Software Name] is licensed under Mulan PSL v2.
|
||||
You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
You may obtain a copy of Mulan PSL v2 at:
|
||||
http://license.coscl.org.cn/MulanPSL2
|
||||
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
||||
See the Mulan PSL v2 for more details.
|
116
qtfs/README.md
Normal file
116
qtfs/README.md
Normal file
@ -0,0 +1,116 @@
|
||||
# qtfs
|
||||
|
||||
## 介绍
|
||||
|
||||
qtfs是一个共享文件系统项目,可部署在host-dpu的硬件架构上,也可以部署在host-vm或同一台host的vm-vm之间,通过vsock建立安全通信通道。以客户端服务器的模式工作,使客户端能通过qtfs访问服务端的指定文件系统,就像访问本地文件系统一样。
|
||||
|
||||
qtfs的特性:
|
||||
+ 支持挂载点传播;
|
||||
+ 支持proc、sys、cgroup等特殊文件系统的共享;
|
||||
+ 客户端对qtfs目录下文件的操作都被转移到服务端,文件读写可共享;
|
||||
+ 支持在客户端对服务端的文件系统进行远程挂载;
|
||||
+ 可以定制化处理特殊文件;
|
||||
+ 支持远端fifo、unix-socket等,并且支持epoll,使客户端和服务端像本地通信一样使用这些文件;
|
||||
+ 基于host-dpu架构时,底层通信方式可以支持PCIe,性能大大优于网络;
|
||||
+ 内核模块形式开发,无需对内核进行侵入式修改。
|
||||
|
||||
## 软件架构
|
||||
|
||||
软件大体框架图:
|
||||
|
||||
![输入图片说明](doc/%20Overall_architecture_diagram.png)
|
||||
|
||||
|
||||
## 安装教程
|
||||
|
||||
目录说明:
|
||||
+ **rexec**:跨主机二进制生命周期管理组件,在该目录下编译rexec和rexec_server。
|
||||
+ **ipc**: 跨主机unix domain socket协同组件,在该目录下编译udsproxyd二进制和libudsproxy.so库。
|
||||
+ **qtfs**: 客户端内核模块相关代码,直接在该目录下编译客户端ko。
|
||||
+ **qtfs_server**: 服务端内核模块相关代码,直接在该目录下编译服务端ko和相关程序。
|
||||
+ **qtinfo**: 诊断工具,支持查询文件系统的工作状态以及修改log级别等。
|
||||
+ **demo**、**test**、**doc**: 测试程序、演示程序以及项目资料等。
|
||||
+ 根目录: 是客户端与服务端都能用到的公共模块代码。
|
||||
|
||||
### VSOCK通信模式
|
||||
|
||||
如有DPU硬件支持通过vsock与host通信,可选择此方法。
|
||||
如果没有硬件,也可以选择host-vm作为qtfs的client与server进行模拟测试,通信通道为vsock:
|
||||
|
||||
1. 启动vm时为vm配置vsock通道,vm可参考如下配置,将vsock段加在devices配置内:
|
||||
```
|
||||
<devices>
|
||||
...
|
||||
<vsock model='virtio'>
|
||||
<cid auto='no' address='10'/>
|
||||
<alias name='vsock0'/>
|
||||
<address type='pci' domain='0x0000' bus='0x05' slot='0x00' function='0x0'/>
|
||||
</vsock>
|
||||
...
|
||||
</devices>
|
||||
```
|
||||
2. 要求内核版本在5.10或更高版本。
|
||||
3. 安装内核开发包:yum install kernel-devel json-c-devel。
|
||||
|
||||
服务端安装:
|
||||
|
||||
1. cd qtfs_server
|
||||
2. make clean && make -j
|
||||
3. insmod qtfs_server.ko qtfs_server_vsock_cid=2 qtfs_server_vsock_port=12345 qtfs_log_level=WARN
|
||||
4. 配置白名单,将qtfs/config/qtfs/whitelist文件拷贝至/etc/qtfs/下,请手动配置需要的白名单选项,至少需要配置一个Mount白名单才能启动后续服务。
|
||||
5. nohup ./engine 16 1 2 12121 10 12121 2>&1 &
|
||||
Tips: 这里的cid需要根据配置决定,如果host作为server端,则cid固定配置为2,如果vm作为server端,则需要配置为前面xml中的cid字段,本例中为10。
|
||||
|
||||
客户端安装:
|
||||
|
||||
1. cd qtfs
|
||||
2. make clean && make -j
|
||||
3. insmod qtfs.ko qtfs_server_vsock_cid=2 qtfs_server_vsock_port=12345 qtfs_log_level=WARN
|
||||
4. cd ../ipc/
|
||||
5. make clean && make && make install
|
||||
6. nohup udsproxyd 1 10 12121 2 12121 2>&1 &
|
||||
Tips:这里插入ko的cid和port配置为与server端一致即可,udsproxyd的cid+port与server端交换位置。
|
||||
|
||||
其他注意事项:
|
||||
|
||||
1. udsproxyd目前也支持vsock和测试模式两种,使用vsock模式时,不能带UDS_TEST_MODE=1进行编译。
|
||||
2. 如果vsock不通,需要检查host是否插入了vhost_vsock内核模块:modprobe vhost_vsock。
|
||||
|
||||
### 测试模式,仅用于测试环境:
|
||||
|
||||
找两台服务器(或虚拟机)配置内核编译环境:
|
||||
|
||||
1. 要求内核版本在5.10或更高版本。
|
||||
2. 安装内核开发包:yum install kernel-devel。
|
||||
3. 假设host服务器ip为192.168.10.10,dpu为192.168.10.11
|
||||
|
||||
服务端安装:
|
||||
|
||||
1. cd qtfs_server
|
||||
2. make clean && make -j QTFS_TEST_MODE=1
|
||||
3.指定测试服务端的ip,ip a a ip_server(例:192.168.10.10)/port dev network(例:ens32),防止机器重启造成的ip变更问题,方便测试
|
||||
4. insmod qtfs_server.ko qtfs_server_ip=x.x.x.x qtfs_server_port=12345 qtfs_log_level=WARN
|
||||
5. 配置白名单,将qtfs/config/qtfs/whitelist文件拷贝至/etc/qtfs/下,请手动配置需要的白名单选项,至少需要配置一个Mount白名单才能启动后续服务。
|
||||
6. nohup ./engine 16 1 192.168.10.10 12121 192.168.10.11 12121 2>&1 &
|
||||
Tips: 该模式暴露网络端口,有可能造成安全隐患,仅能用于功能验证测试,勿用于实际生产环境。
|
||||
|
||||
客户端安装:
|
||||
|
||||
1. cd qtfs
|
||||
2. make clean && make -j QTFS_TEST_MODE=1
|
||||
3.指定测试用户端的ip,ip a a ip_client(例:192.168.10.11)/port dev network(例:ens32),防止机器重启造成的ip变更问题,方便测试
|
||||
3. insmod qtfs.ko qtfs_server_ip=x.x.x.x qtfs_server_port=12345 qtfs_log_level=WARN
|
||||
4. cd ../ipc/
|
||||
5. make clean && make && make install
|
||||
6. nohup udsproxyd 1 192.168.10.11 12121 192.168.10.10 12121 2>&1 &
|
||||
Tips: 该模式暴露网络端口,有可能造成安全隐患,仅能用于功能验证测试,勿用于实际生产环境。
|
||||
|
||||
## 使用说明
|
||||
|
||||
安装完成后,客户端通过挂载把服务端的文件系统让客户端可见,例如:
|
||||
|
||||
mount -t qtfs /home /root/mnt/
|
||||
|
||||
客户端进入"/root/mnt"后便可查看到server端/home目录下的所有文件,以及对其进行相关操作。此操作受到白名单的控制,需要挂载路径在server端白名单的Mount列表,或者在其子目录下,且后续的查看或读写操作都需要开放对应的白名单项才能进行。
|
||||
Tips:若完成测试环境的配置后,无法通过服务端访问所挂载的客户端文件,可检查是否由防火墙的阻断导致。
|
||||
|
35
qtfs/config/qtfs/whitelist
Normal file
35
qtfs/config/qtfs/whitelist
Normal file
@ -0,0 +1,35 @@
|
||||
[Open]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Write]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Read]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Readdir]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Mkdir]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Rmdir]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Create]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Unlink]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Rename]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Setattr]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Setxattr]
|
||||
Path=/var/lib/libvirt/qemu
|
||||
|
||||
[Mount]
|
||||
Path=/var/lib/libvirt
|
1
qtfs/config/rexec/whitelist
Normal file
1
qtfs/config/rexec/whitelist
Normal file
@ -0,0 +1 @@
|
||||
/usr/bin/qemu-kvm
|
12
qtfs/demo/Makefile
Normal file
12
qtfs/demo/Makefile
Normal file
@ -0,0 +1,12 @@
|
||||
CFLAGS=-g -O2
|
||||
|
||||
all: cfifo_r cfifo_w
|
||||
|
||||
cfifo_r: cfifo_r.c
|
||||
gcc $(CFLAGS) -o $@ $^
|
||||
|
||||
cfifo_w: cfifo_w.c
|
||||
gcc $(CFLAGS) -o $@ $^
|
||||
|
||||
clean:
|
||||
@rm -f *.o cfifo_r cfifo_w
|
98
qtfs/demo/cfifo_r.c
Normal file
98
qtfs/demo/cfifo_r.c
Normal file
@ -0,0 +1,98 @@
|
||||
#include <stdio.h>
|
||||
#include <fcntl.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <sys/epoll.h>
|
||||
#define BUF_MAX 256
|
||||
int single_read(int argc, char *argv[])
|
||||
{
|
||||
char buf[BUF_MAX];
|
||||
char *fifo = argv[1];
|
||||
int rfd = open(fifo, O_RDONLY);
|
||||
if (rfd < 0) {
|
||||
printf("open file %s failed.\n", fifo);
|
||||
return 0;
|
||||
}
|
||||
|
||||
do {
|
||||
memset(buf, 0, BUF_MAX);
|
||||
int ret = read(rfd, buf, BUF_MAX);
|
||||
if (ret == -1) {
|
||||
printf("read failed.\n");
|
||||
break;
|
||||
}
|
||||
printf("%s", buf);
|
||||
} while (strcmp(buf, "exit") != 0);
|
||||
close(rfd);
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define MAX_READ_LEN 65536
|
||||
int my_epoll_read(int argc, char *argv[])
|
||||
{
|
||||
int *fd = (int *)malloc((argc - 1) * sizeof(int));
|
||||
char *buf = (char *)malloc(MAX_READ_LEN);
|
||||
|
||||
for (int i = 1; i < argc; i++) {
|
||||
fd[i-1] = open(argv[i], O_RDONLY|O_NONBLOCK);
|
||||
if (fd[i-1] < 0){
|
||||
printf("open file %s failed.\n", argv[i]);
|
||||
return 0;
|
||||
}
|
||||
printf("my epoll read open file:%s success, fd:%d.\n", argv[i], fd[i-1]);
|
||||
}
|
||||
|
||||
struct epoll_event evt;
|
||||
struct epoll_event *evts;
|
||||
int epfd = epoll_create1(0);
|
||||
if (epfd == -1) {
|
||||
printf("epoll create failed.\n");
|
||||
abort();
|
||||
}
|
||||
for (int i = 0; i < argc - 1; i++) {
|
||||
evt.data.fd = fd[i];
|
||||
evt.events = EPOLLIN;
|
||||
int s = epoll_ctl(epfd, EPOLL_CTL_ADD, fd[i], &evt);
|
||||
if (s == -1) {
|
||||
printf("epoll ctl failed, fd:%d\n", fd[i]);
|
||||
abort();
|
||||
}
|
||||
printf("my epoll read epoll ctl fd:%d events:%x success.\n", fd[i], evt.events);
|
||||
}
|
||||
evts = calloc(64, sizeof(evt));
|
||||
while (1) {
|
||||
int n = epoll_wait(epfd, evts, 64, -1);
|
||||
printf("epoll wait get new %d events.\n", n);
|
||||
for (int i = 0; i < n; i++) {
|
||||
int ret;
|
||||
FILE *fp;
|
||||
printf(" > epoll wait new events, cur:%d key:%x data:%lx n:%d.\n", i, evts[i].events, evts[i].data, n);
|
||||
memset(buf, 0, MAX_READ_LEN);
|
||||
if (evts[i].events & EPOLLHUP) {
|
||||
epoll_ctl(epfd, EPOLL_CTL_DEL, evts[i].data.fd, NULL);
|
||||
continue;
|
||||
}
|
||||
fp = fdopen(evts[i].data.fd, "r");
|
||||
fseek(fp, 0, SEEK_SET);
|
||||
ret = read(evts[i].data.fd, buf, MAX_READ_LEN);
|
||||
if (ret <= 0) {
|
||||
printf(" >read fd:%d ret:%d data error.\n", evts[i].data.fd, ret);
|
||||
goto end;
|
||||
} else {
|
||||
printf(" >read fd:%d ret:%d data:%s.\n", evts[i].data.fd, ret, buf);
|
||||
}
|
||||
}
|
||||
}
|
||||
end:
|
||||
close(epfd);
|
||||
for (int i = 0; i < argc-1; i++) {
|
||||
close(fd[i]);
|
||||
}
|
||||
return 1;
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
return my_epoll_read(argc, argv);
|
||||
}
|
||||
|
30
qtfs/demo/cfifo_w.c
Normal file
30
qtfs/demo/cfifo_w.c
Normal file
@ -0,0 +1,30 @@
|
||||
#include <stdio.h>
|
||||
#include <fcntl.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#define BUF_MAX 256
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
char buf[BUF_MAX];
|
||||
char *fifo = argv[1];
|
||||
int wfd = open(fifo, O_WRONLY);
|
||||
if (wfd < 0) {
|
||||
printf("open file %s failed.\n", fifo);
|
||||
return 0;
|
||||
}
|
||||
|
||||
do {
|
||||
int ret;
|
||||
int len;
|
||||
memset(buf, 0, BUF_MAX);
|
||||
fgets(buf, BUF_MAX, stdin);
|
||||
len = strlen(buf);
|
||||
ret = write(wfd, buf, BUF_MAX);
|
||||
if (ret == -1) {
|
||||
break;
|
||||
}
|
||||
} while (strcmp(buf, "exit") != 0);
|
||||
|
||||
close(wfd);
|
||||
return 0;
|
||||
}
|
BIN
qtfs/doc/ Overall_architecture_diagram.png
Normal file
BIN
qtfs/doc/ Overall_architecture_diagram.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 48 KiB |
BIN
qtfs/doc/figures/offload-arch.png
Normal file
BIN
qtfs/doc/figures/offload-arch.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 38 KiB |
BIN
qtfs/doc/figures/qtfs-arch.png
Normal file
BIN
qtfs/doc/figures/qtfs-arch.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 48 KiB |
11
qtfs/doc/overview.md
Normal file
11
qtfs/doc/overview.md
Normal file
@ -0,0 +1,11 @@
|
||||
# 容器管理面DPU无感卸载指南
|
||||
|
||||
本文档介绍基于openEuler操作系统的容器管理面DPU无感卸载功能特性及安装部署方法,该特性可以通过操作系统提供的统一抽象层,屏蔽容器管理面跨主机资源访问的差异,实现容器管理面业务无感卸载到DPU上。
|
||||
|
||||
本文档适用于使用openEuler系统并希望了解和使用操作系统内核及容器的社区开发者、开源爱好者以及相关合作伙伴。使用人员需要具备以下经验和技能:
|
||||
|
||||
- 熟悉Linux基本操作
|
||||
|
||||
- 熟悉linux内核文件系统相关基础机制
|
||||
|
||||
- 对kubernetes和docker有一定了解,熟悉docker及kubernetes部署及使用
|
BIN
qtfs/doc/public_sys-resources/icon-note.gif
Normal file
BIN
qtfs/doc/public_sys-resources/icon-note.gif
Normal file
Binary file not shown.
After Width: | Height: | Size: 394 B |
69
qtfs/doc/qtfs共享文件系统架构及使用手册.md
Normal file
69
qtfs/doc/qtfs共享文件系统架构及使用手册.md
Normal file
@ -0,0 +1,69 @@
|
||||
# qtfs
|
||||
|
||||
## 介绍
|
||||
|
||||
qtfs是一个共享文件系统项目,可部署在host-dpu的硬件架构上,也可以部署在2台服务器之间。以客户端服务器的模式工作,使客户端能通过qtfs访问服务端的指定文件系统,得到本地文件访问一致的体验。
|
||||
|
||||
qtfs的特性:
|
||||
|
||||
+ 支持挂载点传播;
|
||||
|
||||
+ 支持proc、sys、cgroup等特殊文件系统的共享;
|
||||
|
||||
+ 支持远程文件读写的共享;
|
||||
|
||||
+ 支持在客户端对服务端的文件系统进行远程挂载;
|
||||
|
||||
+ 支持特殊文件的定制化处理;
|
||||
|
||||
+ 支持远端fifo、unix-socket等,并且支持epoll,使客户端和服务端像本地通信一样使用这些文件;
|
||||
|
||||
+ 支持基于host-dpu架构通过PCIe协议底层通信,性能大大优于网络;
|
||||
|
||||
+ 支持内核模块形式开发,无需对内核进行侵入式修改。
|
||||
|
||||
## 软件架构
|
||||
|
||||
软件大体框架图:
|
||||
|
||||
![qtfs-arch](./figures/qtfs-arch.png)
|
||||
|
||||
## 安装教程
|
||||
|
||||
目录说明:
|
||||
|
||||
+ **qtfs**: 客户端内核模块相关代码,直接在该目录下编译客户端ko。
|
||||
|
||||
+ **qtfs_server**: 服务端内核模块相关代码,直接在该目录下编译服务端ko和相关程序。
|
||||
|
||||
+ **qtinfo**: 诊断工具,支持查询文件系统的工作状态以及修改log级别等。
|
||||
|
||||
+ **demo**、**test**、**doc**: 测试程序、演示程序以及项目资料等。
|
||||
|
||||
+ 根目录: 客户端与服务端通用的公共模块代码。
|
||||
|
||||
首先找两台服务器(或虚拟机)配置内核编译环境:
|
||||
|
||||
1. 要求内核版本在5.10或更高版本。
|
||||
2. 安装内核开发包:yum install kernel-devel。
|
||||
|
||||
服务端安装:
|
||||
|
||||
1. cd qtfs_server
|
||||
2. make clean && make
|
||||
3. insmod qtfs_server.ko qtfs_server_ip=x.x.x.x qtfs_server_port=12345 qtfs_log_level=WARN
|
||||
4. ./engine 4096 16
|
||||
|
||||
客户端安装:
|
||||
|
||||
1. cd qtfs
|
||||
2. make clean && make
|
||||
3. insmod qtfs.ko qtfs_server_ip=x.x.x.x qtfs_server_port=12345 qtfs_log_level=WARN
|
||||
|
||||
## 使用说明
|
||||
|
||||
安装完成后,客户端通过挂载把服务端的文件系统让客户端可见,例如:
|
||||
|
||||
mount -t qtfs / /root/mnt/
|
||||
|
||||
客户端进入"/root/mnt"后便可查看到server端的所有文件,以及对其进行相关操作。
|
31
qtfs/doc/容器管理面无感卸载.md
Normal file
31
qtfs/doc/容器管理面无感卸载.md
Normal file
@ -0,0 +1,31 @@
|
||||
# 容器管理面无感卸载介绍
|
||||
|
||||
## 概述
|
||||
|
||||
在数据中心及云场景下,随着摩尔定律失效,通用处理单元CPU算力增长速率放缓,而同时网络IO类速率及性能不断攀升,二者增长速率差异形成的剪刀差,即当前通用处理器的处理能力无法跟上网络、磁盘等IO处理的需求。传统数据中心下越来越多的通用CPU算力被IO及管理面等占用,这部分资源损耗称之为数据中心税(Data-center Tax)。据AWS统计,数据中心税可能占据数据中心算力的30%以上,部分场景下甚至可能更多。
|
||||
|
||||
DPU的出现就是为了将这部分算力资源从主机CPU上解放出来,通过将管理面、网络、存储、安全等能力卸载到专有的处理器芯片(DPU)上进行处理加速,达成降本增效的结果。目前主流云厂商如AWS、阿里云、华为云都通过自研芯片完成管理面及相关数据面的卸载,达成数据中心计算资源100%售卖给客户。
|
||||
|
||||
管理面进程卸载到DPU可以通过对组件源码进行拆分达成,将源码根据功能逻辑拆分成独立运行的两部分,分别运行在主机和DPU,达成组件卸载的目的。但是这种做法有以下问题:一是影响组件的软件兼容性,组件后续版本升级和维护需要自己维护相关patch,带来一定的维护工作量;二是卸载工作无法被其他组件继承,后续组件卸载后仍需要进行代码逻辑分析和拆分等工作。为解决上述问题,本方案提出DPU的无感卸载,通过OS提供的抽象层,屏蔽应用在主机和DPU间跨主机访问的差异,让业务进程近似0改动达成卸载到DPU运行的目标,且这部分工作属于操作系统通用层,与上层业务无关,其他业务进行DPU卸载时也可以继承。
|
||||
|
||||
## 架构介绍
|
||||
|
||||
#### 容器管理面DPU无感卸载架构
|
||||
|
||||
**图1**容器管理面DPU无感卸载架构
|
||||
|
||||
![offload-arch](./figures/offload-arch.png)
|
||||
|
||||
如图1所示,容器管理面卸载后,dockerd、kubelet等管理进程运行在DPU侧,容器进程本身运行在HOST,进程之间的交互关系由系统层提供对应的能力来保证:
|
||||
|
||||
* 通信层:DPU和主机之间可能通过PCIe或网络进行通信,需要基于底层物理连接提供通信接口层,为上层业务提供通信接口。
|
||||
|
||||
* 内核共享文件系统qtfs:容器管理面组件kubelet、dockerd与容器进程之间的主要交互通过文件系统进行;管理面工具需要为容器进程准备rootfs、volume等数据面路径;还需要在运行时通过proc文件系统、cgroup文件系统等控制和监控容器进程的资源及状态。共享文件系统的详细介绍参考[共享文件系统介绍](qtfs共享文件系统架构及使用手册.md)
|
||||
|
||||
* 用户态卸载环境:用户态需要使用qtfs为容器管理面准备卸载后的运行时环境,将主机的容器管理及运行时相关目录远程挂载到DPU;另外由于需要挂载proc、sys、cgroup等系统管理文件系统,为防止对DPU原生系统功能的破坏,上述挂载动作都在chroot环境内完成。另外管理面(运行于DPU)和容器进程(运行于主机)之间仍存在调用关系,需要通过远程二进制执行工具(rexec)提供对应功能。
|
||||
|
||||
容器管理面无感卸载的操作步骤可参考[部署指导文档](./无感卸载部署指导.md)
|
||||
|
||||
> ![](./public_sys-resources/icon-note.gif)**说明**:
|
||||
>
|
||||
> 上述操作指导涉及对容器管理面组件的少量改动和rexec工具修改,这些修改基于指定版本,其他版本可基于实际执行环境做适配修改。文档中提供的patch仅供验证指导使用,不具备实际商用的条件
|
166
qtfs/doc/无感卸载部署指导.md
Normal file
166
qtfs/doc/无感卸载部署指导.md
Normal file
@ -0,0 +1,166 @@
|
||||
|
||||
# 容器管理面无感卸载部署指导
|
||||
|
||||
> ![](./public_sys-resources/icon-note.gif)**说明**:
|
||||
>
|
||||
> 本指导涉及对容器管理面组件的少量改动和rexec工具修改,这些修改基于指定版本,其他版本可基于实际执行环境做适配修改。文档中提供的patch仅供验证指导使用,不具备实际商用的条件。
|
||||
|
||||
> ![](./public_sys-resources/icon-note.gif)**说明**:
|
||||
>
|
||||
> 当前共享文件系统之间通信通过网络完成,可通过网络互连的两台物理机器或VM模拟验证。
|
||||
>
|
||||
> 建议用户验证前先搭建可正常使用的kubernetes集群和容器运行环境,针对其中单个节点的管理面进程进行卸载验证,卸载环境(DPU)可选择一台具备网络连接的物理机或VM。
|
||||
|
||||
## 简介
|
||||
|
||||
容器管理面,即kubernetes、dockerd、containerd、isulad等容器的管理工具,而容器管理面卸载,即是将容器管理面卸载到与容器所在机器(以下称为HOST)之外的另一台机器(当前场景下是指DPU,一个具备独立运行环境的硬件集合)上运行。
|
||||
|
||||
我们使用共享文件系统qtfs将HOST上与容器运行相关的目录挂载到DPU上,使得容器管理面工具(运行在DPU)可以访问到这些目录,并为容器(运行在HOST)准备运行所需要的环境,此处,因为需要挂载远端的proc和sys等特殊文件系统,所以,我们创建了一个专门的rootfs以作为kubernetes、dockerd的运行环境(以下称为`/another_rootfs`)。
|
||||
|
||||
并且通过rexec执行容器的拉起、删除等操作,使得可以将容器管理面和容器分离在不同的两台机器上,远程对容器进行管理。
|
||||
|
||||
## 相关组件补丁介绍
|
||||
|
||||
#### rexec介绍
|
||||
|
||||
rexec是一个用go语言开发的远程执行工具,基于docker/libchan下的[rexec](https://github.com/docker/libchan/tree/master/examples/rexec)示例工具改造而成,实现远程调用远端二进制的功能,为方便使用在rexec中增加了环境变量传递和监控原进程退出等能力。
|
||||
|
||||
rexec工具的具体使用方式为在服务器端用`CMD_NET_ADDR=tcp://0.0.0.0:<端口号> rexec_server`的方式拉起rexec服务进程,然后在客户端用`CMD_NET_ADDR=tcp://<服务端ip>:<端口号> rexec [要执行的指令] `的方式启动,便可以调用rexec_server执行需要执行的指令,并等待指令执行结果返回。
|
||||
|
||||
#### dockerd相关改动介绍
|
||||
|
||||
对dockerd的改动基于18.09版本。
|
||||
|
||||
在containerd中,暂时注释掉了通过hook调用libnetwork-setkey的部分,此处不影响容器的拉起。并且,为了docker load的正常使用,注释掉了在mounter_linux.go 中mount函数中一处错误的返回。
|
||||
|
||||
最后,因为在容器管理面的运行环境中,将`/proc`挂在了服务端的proc文件系统,而本地的proc文件系统则挂载在了`/local_proc`,所以,dockerd以及containerd中的对`/proc/self/xxx`或者`/proc/getpid()/xxx`或者相关的文件系统访问的部分,我们统统将`/proc`改为了`/local_proc`。
|
||||
|
||||
#### containerd相关改动介绍
|
||||
|
||||
对于containerd的改动基于containerd-1.2-rc.1版本。
|
||||
|
||||
在获取mountinfo时,因为`/proc/self/mountinfo`只能获取到dockerd本身在本地的mountinfo,而无法获取到服务端的mountinfo,所以,将其改为了`/proc/1/mountinfo`,使其通过获取服务端1号进程mountinfo的方式得到服务端的mountinfo。
|
||||
|
||||
在contaienrd-shim中,将与containerd通信的unix socket改为了用tcp通信,containerd通过`SHIM_HOST`环境变量获取containerd-shim所运行环境的ip,即服务端ip。用shim的哈希值计算出一个端口号,并以此作为通信的端口,来拉起containerd-shim.
|
||||
|
||||
并且,将原来的通过系统调用给contaienr-shim发信号的方式,改为了通过远程调用kill指令的方式向shim发信号,确保了docker杀死容器的行为可以正确的执行。
|
||||
|
||||
#### kubernetes相关改动介绍
|
||||
|
||||
kubelet暂不需要功能性改动,可能会遇到容器QoS管理器首次设置失败的错误,该错误不影响后续Pods拉起流程,暂时忽略该报错。
|
||||
|
||||
## 容器管理面卸载操作指南
|
||||
|
||||
在服务器端和客户端,都要拉起rexec_server。服务器端拉起rexec_server,主要是用于客户端创建容器时用rexec拉起containerd-shim,而客户端拉起rexec_server,则是为了执行containerd-shim对dockerd和containerd的调用。
|
||||
|
||||
#### 服务器端
|
||||
|
||||
创建容器管理面所需要的文件夹,然后插入qtfs_server.ko,并拉起engine进程。
|
||||
|
||||
此外在服务器端,还需要创建rexec脚本/usr/bin/dockerd.
|
||||
|
||||
``` shell
|
||||
#!/bin/bash
|
||||
CMD_NET_ADDR=tcp://<客户端ip>:<rexec端口号> rexec /usr/bin/dockerd $*
|
||||
```
|
||||
|
||||
#### 客户端
|
||||
|
||||
需要准备一个rootfs,作为dockerd与containerd的运行环境,通过如下的脚本,将dockerd、containerd所需要的服务端目录挂载到客户端。并且,需要确保在以下脚本中被挂载的远程目录在服务端和客户端都存在。
|
||||
|
||||
``` shell
|
||||
#!/bin/bash
|
||||
mkdir -p /another_rootfs/var/run/docker/containerd
|
||||
iptables -t nat -N DOCKER
|
||||
echo "---------insmod qtfs ko----------"
|
||||
insmod /YOUR/QTFS/PATH/qtfs.ko qtfs_server_ip=<服务端ip> qtfs_log_level=INFO
|
||||
|
||||
# chroot环境内的proc使用DPU的proc共享文件系统替换,需要将本机真实proc文件系统挂载到local_proc下使用
|
||||
mount -t proc proc /another_rootfs/local_proc/
|
||||
|
||||
# 将chroot内环境与外部环境bind,方便进行配置和运行
|
||||
mount --bind /var/run/ /another_rootfs/var/run/
|
||||
mount --bind /var/lib/ /another_rootfs/var/lib/
|
||||
mount --bind /etc /another_rootfs/etc
|
||||
|
||||
mkdir -p /another_rootfs/var/lib/isulad
|
||||
|
||||
# 在chroot环境内创建并挂载dev、sys和cgroup文件系统
|
||||
mount -t devtmpfs devtmpfs /another_rootfs/dev/
|
||||
mount -t sysfs sysfs /another_rootfs/sys
|
||||
mkdir -p /another_rootfs/sys/fs/cgroup
|
||||
mount -t tmpfs tmpfs /another_rootfs/sys/fs/cgroup
|
||||
list="perf_event freezer files net_cls,net_prio hugetlb pids rdma cpu,cpuacct memory devices blkio cpuset"
|
||||
for i in $list
|
||||
do
|
||||
echo $i
|
||||
mkdir -p /another_rootfs/sys/fs/cgroup/$i
|
||||
mount -t cgroup cgroup -o rw,nosuid,nodev,noexec,relatime,$i /another_rootfs/sys/fs/cgroup/$i
|
||||
done
|
||||
|
||||
## common system dir
|
||||
mount -t qtfs -o proc /proc /another_rootfs/proc
|
||||
echo "proc"
|
||||
mount -t qtfs /sys /another_rootfs/sys
|
||||
echo "cgroup"
|
||||
|
||||
# 挂载容器管理面所需要的共享目录
|
||||
mount -t qtfs /var/lib/docker/containers /another_rootfs/var/lib/docker/containers
|
||||
mount -t qtfs /var/lib/docker/containerd /another_rootfs/var/lib/docker/containerd
|
||||
mount -t qtfs /var/lib/docker/overlay2 /another_rootfs/var/lib/docker/overlay2
|
||||
mount -t qtfs /var/lib/docker/image /another_rootfs/var/lib/docker/image
|
||||
mount -t qtfs /var/lib/docker/tmp /another_rootfs/var/lib/docker/tmp
|
||||
mkdir -p /another_rootfs/run/containerd/io.containerd.runtime.v1.linux/
|
||||
mount -t qtfs /run/containerd/io.containerd.runtime.v1.linux/ /another_rootfs/run/containerd/io.containerd.runtime.v1.linux/
|
||||
mkdir -p /another_rootfs/var/run/docker/containerd
|
||||
mount -t qtfs /var/run/docker/containerd /another_rootfs/var/run/docker/containerd
|
||||
mount -t qtfs /var/lib/kubelet/pods /another_rootfs/var/lib/kubelet/pods
|
||||
```
|
||||
|
||||
在/another_rootfs中,需要创建以下脚本,用来支持部分跨主机操作。
|
||||
|
||||
* /another_rootfs/usr/local/bin/containerd-shim
|
||||
|
||||
``` shell
|
||||
#!/bin/bash
|
||||
CMD_NET_ADDR=tcp://<服务端ip>:<rexec端口号> /usr/bin/rexec /usr/bin/containerd-shim $*
|
||||
```
|
||||
|
||||
* /another_rootfs/usr/local/bin/remote_kill
|
||||
|
||||
``` shell
|
||||
#!/bin/bash
|
||||
CMD_NET_ADDR=tcp://<服务端ip>:<rexec端口号> /usr/bin/rexec /usr/bin/kill $*
|
||||
```
|
||||
|
||||
* /another_rootfs/usr/sbin/modprobe
|
||||
``` shell
|
||||
#!/bin/bash
|
||||
CMD_NET_ADDR=tcp://<服务端ip>:<rexec端口号> /usr/bin/rexec /usr/sbin/modprobe $*
|
||||
```
|
||||
|
||||
在chroot到dockerd和containerd运行所需的rootfs后,用如下的命令拉起dockerd和containerd
|
||||
|
||||
* containerd
|
||||
``` shell
|
||||
#!/bin/bash
|
||||
SHIM_HOST=<服务端ip> containerd --config /var/run/docker/containerd/containerd.toml --address /var/run/containerd/containerd.sock
|
||||
```
|
||||
|
||||
* dockerd
|
||||
``` shell
|
||||
#!/bin/bash
|
||||
SHIM_HOST=<服务端ip> CMD_NET_ADDR=tcp://<服务端ip>:<rexec端口号> /usr/bin/dockerd --containerd /var/run/containerd/containerd.sock
|
||||
```
|
||||
|
||||
* kubelet
|
||||
|
||||
在chroot环境内使用原参数拉起kubelet即可。
|
||||
|
||||
因为我们已经将/var/run/和/another_rootfs/var/run/绑定在了一起,所以可以在正常的rootfs下,通过docker来访问docker.sock接口进行容器管理。
|
||||
|
||||
至此,完成容器管理面卸载到DPU,可以通过docker相关操作进行容器创建、删除等操作,也可以通过kubectl在当前节点进行pods调度和销毁,且实际容器业务进程运行在HOST侧。
|
||||
|
||||
> ![](./public_sys-resources/icon-note.gif)**说明**:
|
||||
>
|
||||
> 本指导所述操作只涉及容器管理面进程卸载,不包含容器网络和数据卷volume等卸载,如有相关需求,需要通过额外的网络或存储卸载能力支持。本指导支持不带网络和存储的容器跨节点拉起。
|
322
qtfs/include/comm.h
Normal file
322
qtfs/include/comm.h
Normal file
@ -0,0 +1,322 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_SERVER_COMM_H__
|
||||
#define __QTFS_SERVER_COMM_H__
|
||||
|
||||
#include <linux/version.h>
|
||||
|
||||
#if (LINUX_VERSION_CODE == KERNEL_VERSION(4,19,90)) || (LINUX_VERSION_CODE == KERNEL_VERSION(4,19,36))
|
||||
#define KVER_4_19 1
|
||||
#endif
|
||||
|
||||
#if (LINUX_VERSION_CODE < KERNEL_VERSION(5,6,0))
|
||||
#define BEFORE_KVER_5_6 1
|
||||
#endif
|
||||
|
||||
#define QTSOCK_WL_MAX_NUM 64
|
||||
#define QTFS_WL_MAX_NUM 64 // QTFS server white list
|
||||
|
||||
#define QTFS_USERP_MAXSIZE 65536
|
||||
|
||||
#define QTFS_MAX_EPEVENTS_NUM 64
|
||||
|
||||
extern struct qtinfo *qtfs_diag_info;
|
||||
|
||||
#define QTFS_CLIENT_DEV "/dev/qtfs_client"
|
||||
#define QTFS_SERVER_DEV "/dev/qtfs_server"
|
||||
|
||||
#define QTFS_IOCTL_MAGIC 'Q'
|
||||
enum {
|
||||
_QTFS_IOCTL_EXEC,
|
||||
_QTFS_IOCTL_THREAD_RUN,
|
||||
_QTFS_IOCTL_EPFDSET,
|
||||
_QTFS_IOCTL_EPOLLT,
|
||||
_QTFS_IOCTL_EPOLL_THREAD_RUN,
|
||||
_QTFS_IOCTL_EXIT,
|
||||
|
||||
_QTFS_IOCTL_ALLINFO,
|
||||
_QTFS_IOCTL_CLEARALL,
|
||||
|
||||
_QTFS_IOCTL_LOG_LEVEL,
|
||||
_QTFS_IOCTL_EPOLL_SUPPORT,
|
||||
|
||||
_QTFS_IOCTL_WL_ADD,
|
||||
_QTFS_IOCTL_WL_DEL,
|
||||
_QTFS_IOCTL_WL_GET,
|
||||
};
|
||||
|
||||
#define QTFS_IOCTL_THREAD_INIT _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_EXEC)
|
||||
#define QTFS_IOCTL_THREAD_RUN _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_THREAD_RUN)
|
||||
#define QTFS_IOCTL_EPFDSET _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_EPFDSET)
|
||||
#define QTFS_IOCTL_EPOLL_THREAD_INIT _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_EPOLLT)
|
||||
#define QTFS_IOCTL_EPOLL_THREAD_RUN _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_EPOLL_THREAD_RUN)
|
||||
#define QTFS_IOCTL_EXIT _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_EXIT)
|
||||
#define QTFS_IOCTL_ALLINFO _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_ALLINFO)
|
||||
#define QTFS_IOCTL_CLEARALL _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_CLEARALL)
|
||||
#define QTFS_IOCTL_LOGLEVEL _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_LOG_LEVEL)
|
||||
#define QTFS_IOCTL_EPOLL_SUPPORT _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_EPOLL_SUPPORT)
|
||||
#define QTFS_IOCTL_WL_ADD _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_WL_ADD)
|
||||
#define QTFS_IOCTL_WL_DEL _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_WL_DEL)
|
||||
#define QTFS_IOCTL_WL_GET _IO(QTFS_IOCTL_MAGIC, _QTFS_IOCTL_WL_GET)
|
||||
|
||||
#define QTINFO_MAX_EVENT_TYPE 36 // look qtreq_type at req.h
|
||||
#define QTFS_FUNCTION_LEN 64
|
||||
|
||||
#define QTFS_MAX_THREADS 16
|
||||
#define QTFS_LOGLEVEL_STRLEN 6
|
||||
|
||||
struct qtfs_server_userp_s {
|
||||
size_t size;
|
||||
void *userp;
|
||||
void *userp2;
|
||||
};
|
||||
|
||||
|
||||
enum {
|
||||
#if defined(QTFS_SERVER) || defined(server)
|
||||
QTFS_WHITELIST_OPEN,
|
||||
QTFS_WHITELIST_WRITE,
|
||||
QTFS_WHITELIST_READ,
|
||||
QTFS_WHITELIST_READDIR,
|
||||
QTFS_WHITELIST_MKDIR,
|
||||
QTFS_WHITELIST_RMDIR,
|
||||
QTFS_WHITELIST_CREATE,
|
||||
QTFS_WHITELIST_UNLINK,
|
||||
QTFS_WHITELIST_RENAME,
|
||||
QTFS_WHITELIST_SETATTR,
|
||||
QTFS_WHITELIST_SETXATTR,
|
||||
QTFS_WHITELIST_MOUNT,
|
||||
QTFS_WHITELIST_KILL,
|
||||
#endif
|
||||
QTFS_WHITELIST_UDSCONNECT,
|
||||
QTFS_WHITELIST_MAX,
|
||||
};
|
||||
|
||||
#define QTFS_PATH_MAX 4096
|
||||
// user-kernel struct
|
||||
struct qtfs_wl_item {
|
||||
unsigned int type : 10, // use to add
|
||||
len : 12, // max len 4096
|
||||
index : 10; // use to query or del (index of items)
|
||||
char *path; // path string
|
||||
};
|
||||
|
||||
struct qtfs_thread_init_s {
|
||||
unsigned int thread_nums;
|
||||
struct qtfs_server_userp_s *userp;
|
||||
};
|
||||
|
||||
struct qtreq_epoll_event {
|
||||
unsigned int events;
|
||||
unsigned long data;
|
||||
};
|
||||
|
||||
struct qtfs_server_epoll_s {
|
||||
int epfd;
|
||||
unsigned int event_nums;
|
||||
struct epoll_event *events;
|
||||
struct epoll_event *kevents;
|
||||
};
|
||||
|
||||
enum qtfs_errcode {
|
||||
QTOK = 0,
|
||||
QTERROR = 1,
|
||||
QTEXIT = 2,
|
||||
};
|
||||
|
||||
struct qtsock_whitelist {
|
||||
unsigned int len;
|
||||
char data[0];
|
||||
};
|
||||
|
||||
// qtinfo start
|
||||
#if (defined(QTFS_CLIENT) || defined(client))
|
||||
enum qtinfo_cnts {
|
||||
QTINF_ACTIV_CONN,
|
||||
QTINF_EPOLL_ADDFDS,
|
||||
QTINF_EPOLL_DELFDS,
|
||||
QTINF_EPOLL_FDERR,
|
||||
QTINF_SEQ_ERR,
|
||||
QTINF_RESTART_SYS,
|
||||
QTINF_TYPE_MISMATCH,
|
||||
QTINF_NUM,
|
||||
};
|
||||
#endif
|
||||
|
||||
#if defined(QTFS_SERVER) || defined(server)
|
||||
enum qtinfo_cnts {
|
||||
QTINF_ACTIV_CONN,
|
||||
QTINF_EPOLL_ADDFDS,
|
||||
QTINF_EPOLL_DELFDS,
|
||||
QTINF_NUM,
|
||||
};
|
||||
#endif
|
||||
|
||||
#if (defined(QTFS_CLIENT) || defined(client) || defined(QTFS_SERVER) || defined(server))
|
||||
// for connection state machine
|
||||
typedef enum {
|
||||
QTCONN_INIT,
|
||||
QTCONN_CONNECTING,
|
||||
QTCONN_ACTIVE,
|
||||
} qtfs_conn_state_e;
|
||||
|
||||
struct qtinfo_client {
|
||||
unsigned long cnts[QTINF_NUM];
|
||||
unsigned long recv_err[QTINFO_MAX_EVENT_TYPE];
|
||||
unsigned long send_err[QTINFO_MAX_EVENT_TYPE];
|
||||
unsigned long i_events[QTINFO_MAX_EVENT_TYPE];
|
||||
unsigned long o_events[QTINFO_MAX_EVENT_TYPE];
|
||||
unsigned long rsp_check[QTINFO_MAX_EVENT_TYPE]; // rsp check err cnts
|
||||
};
|
||||
|
||||
struct qtinfo_server {
|
||||
unsigned long cnts[QTINF_NUM];
|
||||
unsigned long i_events[QTINFO_MAX_EVENT_TYPE];
|
||||
unsigned long o_events[QTINFO_MAX_EVENT_TYPE];
|
||||
unsigned long req_check[QTINFO_MAX_EVENT_TYPE]; // req check err cnts
|
||||
};
|
||||
|
||||
struct qtinfo {
|
||||
union {
|
||||
struct qtinfo_client c;
|
||||
struct qtinfo_server s;
|
||||
};
|
||||
// all struct qtreq_xxx's size
|
||||
unsigned int req_size[QTINFO_MAX_EVENT_TYPE];
|
||||
unsigned int rsp_size[QTINFO_MAX_EVENT_TYPE];
|
||||
int log_level;
|
||||
int thread_state[QTFS_MAX_THREADS];
|
||||
char who_using[QTFS_MAX_THREADS][QTFS_FUNCTION_LEN];
|
||||
int epoll_state;
|
||||
int pvar_vld; // valid param's number
|
||||
int pvar_busy; // busy param's number
|
||||
};
|
||||
|
||||
#define QTINFO_STATE(state) ((state == QTCONN_INIT) ? "INIT" : \
|
||||
((state == QTCONN_CONNECTING) ? "CONNECTING" : \
|
||||
((state == QTCONN_ACTIVE) ? "ACTIVE" : "UNKNOWN")))
|
||||
#endif
|
||||
|
||||
//ko compile
|
||||
#if (defined(QTFS_CLIENT) || defined(client))
|
||||
static inline void qtinfo_clear(void)
|
||||
{
|
||||
int i;
|
||||
for (i = QTINF_SEQ_ERR; i < QTINF_NUM; i++)
|
||||
qtfs_diag_info->c.cnts[i] = 0;
|
||||
memset(qtfs_diag_info->c.recv_err, 0, sizeof(qtfs_diag_info->c.recv_err));
|
||||
memset(qtfs_diag_info->c.send_err, 0, sizeof(qtfs_diag_info->c.send_err));
|
||||
memset(qtfs_diag_info->c.i_events, 0, sizeof(qtfs_diag_info->c.i_events));
|
||||
memset(qtfs_diag_info->c.o_events, 0, sizeof(qtfs_diag_info->c.o_events));
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_cntinc(enum qtinfo_cnts idx)
|
||||
{
|
||||
if (idx >= QTINF_NUM)
|
||||
return;
|
||||
qtfs_diag_info->c.cnts[idx]++;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_cntdec(enum qtinfo_cnts idx)
|
||||
{
|
||||
if (idx >= QTINF_NUM || qtfs_diag_info->c.cnts[idx] == 0)
|
||||
return;
|
||||
qtfs_diag_info->c.cnts[idx]--;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_recvinc(size_t idx)
|
||||
{
|
||||
if (idx >= QTINFO_MAX_EVENT_TYPE)
|
||||
return;
|
||||
qtfs_diag_info->c.i_events[idx]++;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_sendinc(size_t idx)
|
||||
{
|
||||
if (idx >= QTINFO_MAX_EVENT_TYPE)
|
||||
return;
|
||||
qtfs_diag_info->c.o_events[idx]++;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_recverrinc(size_t idx)
|
||||
{
|
||||
if (idx >= QTINFO_MAX_EVENT_TYPE)
|
||||
return;
|
||||
qtfs_diag_info->c.recv_err[idx]++;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_senderrinc(size_t idx)
|
||||
{
|
||||
if (idx >= QTINFO_MAX_EVENT_TYPE)
|
||||
return;
|
||||
qtfs_diag_info->c.send_err[idx]++;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_rspcheckinc(size_t idx)
|
||||
{
|
||||
if (idx >= QTINFO_MAX_EVENT_TYPE)
|
||||
return;
|
||||
qtfs_diag_info->c.rsp_check[idx]++;
|
||||
return;
|
||||
}
|
||||
#endif
|
||||
|
||||
// ko compile
|
||||
#if defined(QTFS_SERVER) || defined(server)
|
||||
static inline void qtinfo_clear(void)
|
||||
{
|
||||
memset(qtfs_diag_info->s.i_events, 0, sizeof(qtfs_diag_info->s.i_events));
|
||||
memset(qtfs_diag_info->s.o_events, 0, sizeof(qtfs_diag_info->s.o_events));
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_cntinc(enum qtinfo_cnts idx)
|
||||
{
|
||||
if (idx >= QTINF_NUM)
|
||||
return;
|
||||
qtfs_diag_info->s.cnts[idx]++;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_cntdec(enum qtinfo_cnts idx)
|
||||
{
|
||||
if (idx >= QTINF_NUM || qtfs_diag_info->s.cnts[idx] == 0)
|
||||
return;
|
||||
qtfs_diag_info->s.cnts[idx]--;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_recvinc(size_t idx)
|
||||
{
|
||||
if (idx >= QTINFO_MAX_EVENT_TYPE)
|
||||
return;
|
||||
qtfs_diag_info->s.i_events[idx]++;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_sendinc(size_t idx)
|
||||
{
|
||||
if (idx >= QTINFO_MAX_EVENT_TYPE)
|
||||
return;
|
||||
qtfs_diag_info->s.o_events[idx]++;
|
||||
return;
|
||||
}
|
||||
static inline void qtinfo_reqcheckinc(size_t idx)
|
||||
{
|
||||
if (idx >= QTINFO_MAX_EVENT_TYPE)
|
||||
return;
|
||||
qtfs_diag_info->s.req_check[idx]++;
|
||||
return;
|
||||
}
|
||||
#endif
|
||||
// QTINFO END
|
||||
|
||||
#endif
|
||||
|
219
qtfs/include/conn.h
Normal file
219
qtfs/include/conn.h
Normal file
@ -0,0 +1,219 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_CONN_H__
|
||||
#define __QTFS_CONN_H__
|
||||
|
||||
#include <linux/socket.h>
|
||||
#include <net/sock.h>
|
||||
#include <linux/in.h>
|
||||
#include <linux/inet.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/mount.h>
|
||||
#include <net/inet_sock.h>
|
||||
|
||||
#include "comm.h"
|
||||
#include "log.h"
|
||||
|
||||
#ifdef QTFS_SERVER
|
||||
extern int qtfs_server_thread_run;
|
||||
extern struct qtfs_server_userp_s *qtfs_userps;
|
||||
#endif
|
||||
extern char qtfs_conn_type[20];
|
||||
extern char qtfs_server_ip[20];
|
||||
extern int qtfs_server_port;
|
||||
extern unsigned int qtfs_server_vsock_port;
|
||||
extern unsigned int qtfs_server_vsock_cid;
|
||||
|
||||
extern struct qtfs_conn_var_s *qtfs_thread_var[QTFS_MAX_THREADS];
|
||||
extern struct qtfs_conn_var_s *qtfs_epoll_var;
|
||||
extern char qtfs_log_level[QTFS_LOGLEVEL_STRLEN];
|
||||
extern int log_level;
|
||||
extern struct qtinfo *qtfs_diag_info;
|
||||
extern bool qtfs_epoll_mode;
|
||||
extern struct qtfs_pvar_ops_s qtfs_conn_sock_pvar_ops;
|
||||
extern struct qtfs_wl g_qtfs_wl;
|
||||
|
||||
struct qtfs_conn_var_s *_qtfs_conn_get_param(const char *);
|
||||
static inline struct qtfs_conn_var_s *__qtfs_conn_get_param(const char *who_using)
|
||||
{
|
||||
struct qtfs_conn_var_s *p = _qtfs_conn_get_param(who_using);
|
||||
if (IS_ERR_OR_NULL(p))
|
||||
return NULL;
|
||||
return p;
|
||||
}
|
||||
#define qtfs_conn_get_param(void) __qtfs_conn_get_param(__func__)
|
||||
#define qtfs_conn_get_param_errcode(void) _qtfs_conn_get_param(__func__)
|
||||
|
||||
#define QTFS_CONN_SOCK_TYPE "socket"
|
||||
#define QTFS_CONN_PCIE_TYPE "pcie"
|
||||
|
||||
#define QTFS_EPOLL_THREADIDX (QTFS_MAX_THREADS + 4)
|
||||
#define QTCONN_IS_EPOLL_CONN(pvar) (pvar->cur_threadidx == QTFS_EPOLL_THREADIDX)
|
||||
#define QTFS_SERVER_MAXCONN 2
|
||||
#define QTFS_GET_PARAM_MAX_RETRY 10000
|
||||
|
||||
static inline long __must_check QTFS_PTR_ERR(__force const void *ptr)
|
||||
{
|
||||
if (!ptr)
|
||||
return -EINVAL;
|
||||
return (long) ptr;
|
||||
}
|
||||
|
||||
static inline bool qtfs_support_epoll(umode_t mode)
|
||||
{
|
||||
return (qtfs_epoll_mode || S_ISFIFO(mode));
|
||||
}
|
||||
|
||||
#define QTFS_SOCK_RCVTIMEO 1
|
||||
#define QTFS_SOCK_SNDTIMEO 1
|
||||
|
||||
typedef enum {
|
||||
QTFS_CONN_SOCKET,
|
||||
QTFS_CONN_PCIE,
|
||||
QTFS_CONN_INVALID,
|
||||
} qtfs_conn_mode_e;
|
||||
|
||||
typedef enum {
|
||||
QTFS_CONN_SOCK_SERVER,
|
||||
QTFS_CONN_SOCK_CLIENT,
|
||||
} qtfs_conn_cs_e;
|
||||
|
||||
// 使用pvar的主题类型,为了更好区分类型,socket下server有主socket下建立多个链接
|
||||
// 的需求,通过类型告知socket模块,且解藕conn层和下面的不同消息通道类型
|
||||
typedef enum {
|
||||
QTFS_CONN_TYPE_QTFS, // QTFS tunnel type
|
||||
QTFS_CONN_TYPE_EPOLL, // EPOLL thread type
|
||||
QTFS_CONN_TYPE_FIFO, // FIFO type
|
||||
QTFS_CONN_TYPE_INV,
|
||||
} qtfs_conn_type_e;
|
||||
|
||||
struct qtfs_pcie_var_s {
|
||||
int srcid;
|
||||
int dstid;
|
||||
};
|
||||
|
||||
struct qtfs_sock_var_s {
|
||||
struct socket *sock;
|
||||
struct socket *client_sock;
|
||||
#ifdef QTFS_TEST_MODE
|
||||
char addr[20];
|
||||
unsigned short port;
|
||||
#else
|
||||
// for vsock
|
||||
unsigned int vm_port;
|
||||
unsigned int vm_cid;
|
||||
#endif
|
||||
};
|
||||
|
||||
struct qtfs_conn_ops_s {
|
||||
// conn message buffer initialization and releasement.
|
||||
int (*conn_var_init)(struct qtfs_conn_var_s *pvar);
|
||||
void (*conn_var_fini)(struct qtfs_conn_var_s *pvar);
|
||||
void (*conn_msg_clear)(struct qtfs_conn_var_s *pvar);
|
||||
void *(*get_conn_msg_buf)(struct qtfs_conn_var_s *pvar, int dir);
|
||||
|
||||
// connection related ops
|
||||
int (*conn_init)(void *connvar, qtfs_conn_type_e type);
|
||||
void (*conn_fini)(void *connvar, qtfs_conn_type_e type);
|
||||
int (*conn_new_connection)(void *connvar, qtfs_conn_type_e type);
|
||||
int (*conn_send)(void *connvar, void *buf, size_t len);
|
||||
int (*conn_recv)(void *connvar, void *buf, size_t len, bool block);
|
||||
int (*conn_send_iter)(void *connvar, struct iov_iter *iov);
|
||||
int (*conn_recv_iter)(void *connvar, struct iov_iter *iov, bool block);
|
||||
bool (*conn_inited)(void *connvar, qtfs_conn_type_e type);
|
||||
bool (*conn_connected)(void *connvar);
|
||||
void (*conn_recv_buff_drop)(void *connvar);
|
||||
};
|
||||
|
||||
struct qtfs_pvar_ops_s {
|
||||
// channel-specific parameter parsing function
|
||||
int (*parse_param)(void);
|
||||
// channel-specific global param init
|
||||
int (*param_init)(void);
|
||||
int (*param_fini)(void);
|
||||
// init pvar with channel specific ops
|
||||
int (*pvar_init)(void *connvar, struct qtfs_conn_ops_s **conn_ops, qtfs_conn_type_e type);
|
||||
};
|
||||
extern struct qtfs_pvar_ops_s *g_pvar_ops;
|
||||
|
||||
struct qtfs_conn_var_s {
|
||||
struct list_head lst;
|
||||
struct llist_node lazy_put;
|
||||
int cur_threadidx;
|
||||
int miss_proc;
|
||||
unsigned long seq_num;
|
||||
qtfs_conn_state_e state;
|
||||
qtfs_conn_type_e user_type; // type of pvar's user: qtfs/epoll deamon/fifo
|
||||
char who_using[QTFS_FUNCTION_LEN];
|
||||
union {
|
||||
struct qtfs_sock_var_s sock_var;
|
||||
struct qtfs_pcie_var_s pcie_var;
|
||||
} conn_var;
|
||||
struct qtfs_conn_ops_s *conn_ops;
|
||||
|
||||
// use to memset buf
|
||||
unsigned int recv_valid;
|
||||
unsigned int send_valid;
|
||||
unsigned int recv_max;
|
||||
unsigned int send_max;
|
||||
struct kvec vec_recv;
|
||||
struct kvec vec_send;
|
||||
struct iov_iter *iov_recv; // for non-copy
|
||||
struct iov_iter *iov_send; // for non-copy
|
||||
unsigned int magic_send;
|
||||
unsigned int magic_recv;
|
||||
};
|
||||
|
||||
struct qtfs_wl_cap {
|
||||
unsigned int nums;
|
||||
char *item[QTFS_WL_MAX_NUM];
|
||||
};
|
||||
|
||||
struct qtfs_wl {
|
||||
rwlock_t rwlock;
|
||||
struct qtfs_wl_cap cap[QTFS_WHITELIST_MAX];
|
||||
};
|
||||
|
||||
int qtfs_conn_init(struct qtfs_conn_var_s *pvar);
|
||||
void qtfs_conn_fini(struct qtfs_conn_var_s *pvar);
|
||||
int qtfs_conn_send(struct qtfs_conn_var_s *pvar);
|
||||
int qtfs_conn_recv(struct qtfs_conn_var_s *pvar);
|
||||
int qtfs_conn_recv_block(struct qtfs_conn_var_s *pvar);
|
||||
|
||||
int qtfs_conn_var_init(struct qtfs_conn_var_s *pvar);
|
||||
void qtfs_conn_var_fini(struct qtfs_conn_var_s *pvar);
|
||||
void qtfs_conn_msg_clear(struct qtfs_conn_var_s *pvar);
|
||||
void *qtfs_conn_msg_buf(struct qtfs_conn_var_s *pvar, int dir);
|
||||
|
||||
int qtfs_conn_param_init(void);
|
||||
void qtfs_conn_param_fini(void);
|
||||
|
||||
void qtfs_conn_put_param(struct qtfs_conn_var_s *pvar);
|
||||
struct qtfs_conn_var_s *qtfs_epoll_establish_conn(void);
|
||||
void qtfs_epoll_cut_conn(struct qtfs_conn_var_s *pvar);
|
||||
|
||||
#ifdef QTFS_CLIENT
|
||||
struct qtfs_conn_var_s *qtfs_fifo_get_param(void);
|
||||
void qtfs_fifo_put_param(struct qtfs_conn_var_s *pvar);
|
||||
#endif
|
||||
|
||||
int qtfs_sm_active(struct qtfs_conn_var_s *pvar);
|
||||
int qtfs_sm_reconnect(struct qtfs_conn_var_s *pvar);
|
||||
int qtfs_sm_exit(struct qtfs_conn_var_s *pvar);
|
||||
|
||||
void qtfs_conn_list_cnt(void);
|
||||
|
||||
int qtfs_uds_remote_connect_user(int fd, struct sockaddr __user *addr, int len);
|
||||
|
||||
#endif
|
126
qtfs/include/log.h
Normal file
126
qtfs/include/log.h
Normal file
@ -0,0 +1,126 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_LOG_H__
|
||||
#define __QTFS_LOG_H__
|
||||
|
||||
#include <linux/string.h>
|
||||
#include "comm.h"
|
||||
|
||||
enum level {
|
||||
LOG_NONE,
|
||||
LOG_ERROR,
|
||||
LOG_WARN,
|
||||
LOG_INFO,
|
||||
LOG_DEBUG
|
||||
};
|
||||
|
||||
extern int log_level;
|
||||
|
||||
#define qtfs_crit(fmt, ...) \
|
||||
{\
|
||||
pr_crit("[%s::%s:%4d] " fmt,\
|
||||
KBUILD_MODNAME, kbasename(__FILE__), __LINE__, ##__VA_ARGS__);\
|
||||
}
|
||||
|
||||
#define qtfs_err(fmt, ...) \
|
||||
( \
|
||||
{ \
|
||||
if (likely(log_level >= LOG_ERROR)) { \
|
||||
pr_err("[%s::%s:%4d] " fmt, \
|
||||
KBUILD_MODNAME, kbasename(__FILE__), __LINE__, ##__VA_ARGS__); \
|
||||
} \
|
||||
} \
|
||||
)
|
||||
|
||||
static inline int qtfs_log_init(char *level, int len) {
|
||||
if (!strcmp(level, "WARN")) {
|
||||
log_level = LOG_WARN;
|
||||
} else if (!strcmp(level, "INFO")) {
|
||||
log_level = LOG_INFO;
|
||||
} else if (!strcmp(level, "DEBUG")) {
|
||||
log_level = LOG_DEBUG;
|
||||
} else if (!strcmp(level, "NONE")) {
|
||||
log_level = LOG_NONE;
|
||||
} else if(!strcmp(level, "ERROR")){
|
||||
log_level = LOG_ERROR;
|
||||
} else {
|
||||
qtfs_err("qtfs log set failed, unknown type:%s.", level);
|
||||
return QTERROR;
|
||||
}
|
||||
return QTOK;
|
||||
}
|
||||
|
||||
|
||||
#define qtfs_warn(fmt, ...) \
|
||||
( \
|
||||
{ \
|
||||
if (unlikely(log_level >= LOG_WARN)) { \
|
||||
pr_warn("[%s::%s:%4d] " fmt, \
|
||||
KBUILD_MODNAME, kbasename(__FILE__), __LINE__, ##__VA_ARGS__); \
|
||||
} \
|
||||
} \
|
||||
)
|
||||
|
||||
#define qtfs_info(fmt, ...) \
|
||||
( \
|
||||
{ \
|
||||
if (unlikely(log_level >= LOG_INFO)) { \
|
||||
pr_info("[%s::%s:%4d] " fmt, \
|
||||
KBUILD_MODNAME, kbasename(__FILE__), __LINE__, ##__VA_ARGS__); \
|
||||
} \
|
||||
} \
|
||||
)
|
||||
|
||||
#define qtfs_debug(fmt, ...) \
|
||||
( \
|
||||
{ \
|
||||
if (unlikely(log_level >= LOG_DEBUG)) { \
|
||||
pr_info("[%s::%s:%4d] " fmt, \
|
||||
KBUILD_MODNAME, kbasename(__FILE__), __LINE__, ##__VA_ARGS__); \
|
||||
} \
|
||||
} \
|
||||
)
|
||||
|
||||
#define qtfs_err_ratelimited(fmt, ...) \
|
||||
( \
|
||||
{ \
|
||||
if (likely(log_level >= LOG_ERROR)) { \
|
||||
pr_err_ratelimited("[%s::%s:%4d] " fmt, \
|
||||
KBUILD_MODNAME, kbasename(__FILE__), __LINE__, ##__VA_ARGS__); \
|
||||
} \
|
||||
} \
|
||||
)
|
||||
|
||||
#define qtfs_info_ratelimited(fmt, ...) \
|
||||
( \
|
||||
{ \
|
||||
if (unlikely(log_level >= LOG_INFO)) { \
|
||||
pr_info_ratelimited("[%s::%s:%4d] " fmt, \
|
||||
KBUILD_MODNAME, kbasename(__FILE__), __LINE__, ##__VA_ARGS__); \
|
||||
} \
|
||||
} \
|
||||
)
|
||||
|
||||
#define qtfs_warn_ratelimited(fmt, ...) \
|
||||
( \
|
||||
{ \
|
||||
if (unlikely(log_level >= LOG_WARN)) { \
|
||||
pr_warn_ratelimited("[%s::%s:%4d] " fmt, \
|
||||
KBUILD_MODNAME, kbasename(__FILE__), __LINE__, ##__VA_ARGS__); \
|
||||
} \
|
||||
} \
|
||||
)
|
||||
|
||||
|
||||
#endif
|
59
qtfs/include/qtfs_check.h
Normal file
59
qtfs/include/qtfs_check.h
Normal file
@ -0,0 +1,59 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_CHECK_H__
|
||||
#define __QTFS_CHECK_H__
|
||||
|
||||
#include "req.h"
|
||||
|
||||
enum {
|
||||
QTFS_CHECK_OK,
|
||||
QTFS_CHECK_ERR,
|
||||
};
|
||||
|
||||
int req_check_none(void *in);
|
||||
int req_check_mount(void *in);
|
||||
int req_check_open(void *in);
|
||||
int req_check_close(void *in);
|
||||
int req_check_readiter(void *in);
|
||||
int req_check_write(void *in);
|
||||
int req_check_lookup(void *in);
|
||||
int req_check_readdir(void *in);
|
||||
int req_check_mkdir(void *in);
|
||||
int req_check_rmdir(void *in);
|
||||
int req_check_getattr(void *in);
|
||||
int req_check_setattr(void *in);
|
||||
int req_check_icreate(void *in);
|
||||
int req_check_mknod(void *in);
|
||||
int req_check_unlink(void *in);
|
||||
int req_check_symlink(void *in);
|
||||
int req_check_link(void *in);
|
||||
int req_check_getlink(void *in);
|
||||
int req_check_readlink(void *in);
|
||||
int req_check_rename(void *in);
|
||||
int req_check_xattrlist(void *in);
|
||||
int req_check_xattrget(void *in);
|
||||
int req_check_xattrset(void *in);
|
||||
int req_check_sysmount(void *in);
|
||||
int req_check_sysumount(void *in);
|
||||
int req_check_fifopoll(void *in);
|
||||
int req_check_statfs(void *in);
|
||||
int req_check_ioctl(void *in);
|
||||
int req_check_epoll_ctl(void *in);
|
||||
int req_check_llseek(void *in);
|
||||
int req_check_sc_kill(void *in);
|
||||
int req_check_sc_sched_getaffinity(void *in);
|
||||
int req_check_sc_sched_setaffinity(void *in);
|
||||
|
||||
#endif
|
||||
|
562
qtfs/include/req.h
Normal file
562
qtfs/include/req.h
Normal file
@ -0,0 +1,562 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_REQ_STRUCT_DEF_H__
|
||||
#define __QTFS_REQ_STRUCT_DEF_H__
|
||||
|
||||
#include <linux/fs.h>
|
||||
#include <linux/statfs.h>
|
||||
#include <uapi/linux/limits.h>
|
||||
#include "comm.h"
|
||||
#include "log.h"
|
||||
|
||||
enum qtreq_type {
|
||||
QTFS_REQ_NULL,
|
||||
QTFS_REQ_MOUNT,
|
||||
QTFS_REQ_OPEN,
|
||||
QTFS_REQ_CLOSE,
|
||||
QTFS_REQ_READ,
|
||||
QTFS_REQ_READITER, // 5
|
||||
QTFS_REQ_WRITE,
|
||||
QTFS_REQ_LOOKUP,
|
||||
QTFS_REQ_READDIR,
|
||||
QTFS_REQ_MKDIR,
|
||||
QTFS_REQ_RMDIR, // 10
|
||||
QTFS_REQ_GETATTR,
|
||||
QTFS_REQ_SETATTR,
|
||||
QTFS_REQ_ICREATE,
|
||||
QTFS_REQ_MKNOD,
|
||||
QTFS_REQ_UNLINK, // 15
|
||||
QTFS_REQ_SYMLINK,
|
||||
QTFS_REQ_LINK,
|
||||
QTFS_REQ_GETLINK,
|
||||
QTFS_REQ_READLINK,
|
||||
QTFS_REQ_RENAME, // 20
|
||||
|
||||
QTFS_REQ_XATTRLIST,
|
||||
QTFS_REQ_XATTRGET,
|
||||
QTFS_REQ_XATTRSET,
|
||||
|
||||
QTFS_REQ_SYSMOUNT,
|
||||
QTFS_REQ_SYSUMOUNT, // 25
|
||||
QTFS_REQ_FIFOPOLL,
|
||||
|
||||
QTFS_REQ_STATFS,
|
||||
QTFS_REQ_IOCTL,
|
||||
|
||||
QTFS_REQ_EPOLL_CTL,
|
||||
|
||||
QTFS_REQ_EPOLL_EVENT, // 30
|
||||
|
||||
QTFS_REQ_LLSEEK,
|
||||
|
||||
// REMOTE SYSCALL
|
||||
QTFS_SC_KILL,
|
||||
QTFS_SC_SCHED_GETAFFINITY,
|
||||
QTFS_SC_SCHED_SETAFFINITY,
|
||||
|
||||
QTFS_REQ_EXIT, // exit server thread
|
||||
QTFS_REQ_INV,
|
||||
};
|
||||
#define QTFS_REQ_TYPEVALID(type) (type < QTFS_REQ_INV && type >= QTFS_REQ_NULL)
|
||||
|
||||
|
||||
enum qtreq_ret {
|
||||
QTFS_OK,
|
||||
QTFS_ERR,
|
||||
};
|
||||
|
||||
enum qtfs_type {
|
||||
QTFS_NORMAL,
|
||||
QTFS_PROC,
|
||||
QTFS_SYS, // for sysfs
|
||||
};
|
||||
|
||||
struct qtfs_dirent64 {
|
||||
u64 d_ino;
|
||||
s64 d_off;
|
||||
unsigned short d_reclen;
|
||||
unsigned char d_type;
|
||||
unsigned char resv[5];
|
||||
char d_name[];
|
||||
};
|
||||
|
||||
#define QTFS_SEND 0
|
||||
#define QTFS_RECV 1
|
||||
|
||||
// maximum possible length, can be increased according to the actual situation
|
||||
#define NAME_MAX 255
|
||||
#define MAX_PATH_LEN PATH_MAX
|
||||
#define MAX_ELSE_LEN (1024 * 128)
|
||||
#define QTFS_REQ_MAX_LEN (MAX_PATH_LEN + MAX_ELSE_LEN)
|
||||
|
||||
#define MAX_BUF 4096
|
||||
|
||||
// QTFS_TAIL_LEN解释:
|
||||
// 私有数据结构最大长度为QTFS_REQ_MAX_LEN,超出就越界了
|
||||
// 一般有变长buf要求的,把变长buf放在末尾
|
||||
// 其长度定义为QTFS_REQ_MAX_LEN减去前面所有成员结构长度
|
||||
// 尾部变长数组长度自定义的,整体结构体长度不能超出最大长度
|
||||
#define QTFS_TAIL_LEN(head) (QTFS_REQ_MAX_LEN - sizeof(head))
|
||||
|
||||
// QTFS_SEND_SIZE解释:
|
||||
// 用来发送的数据结构buf为固定大小QTFS_REQ_MAX_LEN
|
||||
// 但是我们大多数时候只使用了少量的bytes
|
||||
// 只需要发送有效数据,所以私有数据结构一般采取一些关键
|
||||
// 字段,加一个动态buf的组合方式,buf放在结构体末尾
|
||||
// 当传输时,只传输关键字段和动态buf的有效长度,可以用这个宏
|
||||
// 来计算所需发送的有效长度
|
||||
// 如果结构体定义不是:关键字段+字符串buf的模式,则不能用这个宏
|
||||
// 因为这个宏使用了strlen来测量末尾有效长度
|
||||
#define QTFS_SEND_SIZE(stru, tailstr) sizeof(stru) - sizeof(tailstr) + strlen(tailstr) + 1
|
||||
|
||||
struct qtreq {
|
||||
unsigned int type; // operation type
|
||||
unsigned int err;
|
||||
unsigned long seq_num; // check code
|
||||
size_t len;
|
||||
char data[0]; // operation's private data
|
||||
};
|
||||
|
||||
#define QTFS_MSG_LEN sizeof(struct qtreq) + QTFS_REQ_MAX_LEN
|
||||
#define QTFS_MSG_HEAD_LEN sizeof(struct qtreq)
|
||||
|
||||
#define QTFS_REQ_MAGIC 0xa55aa55a
|
||||
|
||||
#define QTFS_FIFO_HEAD_LEN 32 // fifo只用很少的额外头,32应该足够了
|
||||
#define QTFS_FIFO_REQ_LEN (QTFS_MSG_HEAD_LEN + QTFS_FIFO_HEAD_LEN)
|
||||
|
||||
struct qtreq_ioctl {
|
||||
struct qtreq_ioctl_len {
|
||||
unsigned int cmd;
|
||||
unsigned int size;
|
||||
int fd;
|
||||
int argtype; // 0--use pointer arg, 1--use long arg
|
||||
unsigned long arg; // for long type arg
|
||||
} d;
|
||||
|
||||
char path[QTFS_TAIL_LEN(struct qtreq_ioctl_len)];
|
||||
};
|
||||
|
||||
struct qtrsp_ioctl {
|
||||
int ret;
|
||||
int errno;
|
||||
unsigned int size;
|
||||
|
||||
char buf[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtreq_statfs {
|
||||
char path[MAX_PATH_LEN]; // include file name
|
||||
};
|
||||
|
||||
struct qtrsp_statfs {
|
||||
struct kstatfs kstat;
|
||||
int ret;
|
||||
int errno;
|
||||
};
|
||||
|
||||
struct qtreq_mount {
|
||||
char path[MAX_PATH_LEN]; // include file name
|
||||
};
|
||||
struct qtrsp_mount {
|
||||
int ret;
|
||||
int errno;
|
||||
};
|
||||
|
||||
struct qtreq_open {
|
||||
__u64 flags;
|
||||
unsigned int mode;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_open {
|
||||
int fd;
|
||||
int ret;
|
||||
};
|
||||
|
||||
struct qtreq_close {
|
||||
int fd;
|
||||
};
|
||||
|
||||
struct qtrsp_close {
|
||||
int ret;
|
||||
};
|
||||
|
||||
struct qtreq_readiter {
|
||||
size_t len;
|
||||
long long pos;
|
||||
int fd;
|
||||
};
|
||||
|
||||
struct qtrsp_readiter {
|
||||
struct qtrsp_readiter_len {
|
||||
int ret;
|
||||
ssize_t len;
|
||||
int errno;
|
||||
int end;
|
||||
} d;
|
||||
char readbuf[QTFS_TAIL_LEN(struct qtrsp_readiter_len)];
|
||||
};
|
||||
|
||||
struct qtreq_write {
|
||||
struct qtreq_write_len {
|
||||
int buflen;
|
||||
long long pos;
|
||||
int fd;
|
||||
long long flags;
|
||||
long long mode;
|
||||
long long total_len;
|
||||
} d;
|
||||
// fullname and writebuf
|
||||
char path_buf[QTFS_TAIL_LEN(struct qtreq_write_len)];
|
||||
};
|
||||
|
||||
struct qtrsp_write {
|
||||
int ret;
|
||||
ssize_t len; // 成功写入的长度
|
||||
};
|
||||
|
||||
struct qtreq_mmap {
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_mmap {
|
||||
int ret;
|
||||
};
|
||||
|
||||
struct qtreq_lookup {
|
||||
char fullname[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct inode_info {
|
||||
unsigned int mode;
|
||||
unsigned short i_opflags;
|
||||
kuid_t i_uid;
|
||||
kgid_t i_gid;
|
||||
unsigned int i_flags;
|
||||
unsigned long i_ino;
|
||||
|
||||
dev_t i_rdev;
|
||||
long long i_size;
|
||||
|
||||
struct timespec64 atime;
|
||||
struct timespec64 mtime;
|
||||
struct timespec64 ctime;
|
||||
|
||||
unsigned short i_bytes;
|
||||
u8 i_blkbits;
|
||||
u8 i_write_hint;
|
||||
blkcnt_t i_blocks;
|
||||
|
||||
unsigned long i_state;
|
||||
unsigned long dirtied_when; /* jiffies of first dirtying */
|
||||
unsigned long dirtied_time_when;
|
||||
|
||||
__u32 i_generation;
|
||||
};
|
||||
|
||||
struct qtrsp_lookup {
|
||||
int ret;
|
||||
int errno;
|
||||
struct inode_info inode_info;
|
||||
};
|
||||
|
||||
struct qtreq_readdir {
|
||||
int count;
|
||||
loff_t pos;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_readdir {
|
||||
struct qtrsp_readdir_len {
|
||||
int ret;
|
||||
int vldcnt;
|
||||
int over; // 是否已经全部获取完成
|
||||
loff_t pos;
|
||||
} d;
|
||||
char dirent[QTFS_TAIL_LEN(struct qtrsp_readdir_len)];
|
||||
};
|
||||
|
||||
struct qtreq_mkdir {
|
||||
umode_t mode;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_mkdir {
|
||||
int ret;
|
||||
int errno;
|
||||
struct inode_info inode_info;
|
||||
};
|
||||
|
||||
struct qtreq_rmdir {
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_rmdir {
|
||||
int ret;
|
||||
int errno;
|
||||
};
|
||||
|
||||
struct qtreq_getattr {
|
||||
u32 request_mask;
|
||||
unsigned int query_flags;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_getattr {
|
||||
int ret;
|
||||
int errno;
|
||||
struct kstat stat;
|
||||
};
|
||||
|
||||
struct qtreq_setattr {
|
||||
struct iattr attr;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_setattr {
|
||||
int ret;
|
||||
int errno;
|
||||
};
|
||||
|
||||
struct qtreq_icreate {
|
||||
umode_t mode;
|
||||
bool excl;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_icreate {
|
||||
int ret;
|
||||
int errno;
|
||||
struct inode_info inode_info;
|
||||
};
|
||||
|
||||
struct qtreq_mknod {
|
||||
umode_t mode;
|
||||
dev_t dev;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_mknod {
|
||||
int ret;
|
||||
int errno;
|
||||
struct inode_info inode_info;
|
||||
};
|
||||
|
||||
struct qtreq_unlink {
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_unlink {
|
||||
int errno;
|
||||
};
|
||||
|
||||
struct qtreq_symlink {
|
||||
struct qtreq_symlink_len {
|
||||
size_t newlen;
|
||||
size_t oldlen;
|
||||
} d;
|
||||
char path[QTFS_TAIL_LEN(struct qtreq_symlink_len)];
|
||||
};
|
||||
|
||||
struct qtrsp_symlink {
|
||||
int ret;
|
||||
int errno;
|
||||
struct inode_info inode_info;
|
||||
};
|
||||
|
||||
struct qtreq_link {
|
||||
struct qtreq_link_len {
|
||||
size_t newlen;
|
||||
size_t oldlen;
|
||||
} d;
|
||||
char path[QTFS_TAIL_LEN(struct qtreq_link_len)];
|
||||
};
|
||||
|
||||
struct qtrsp_link {
|
||||
int ret;
|
||||
int errno;
|
||||
struct inode_info inode_info;
|
||||
};
|
||||
|
||||
struct qtreq_getlink {
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_getlink {
|
||||
int ret;
|
||||
int errno;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtreq_rename {
|
||||
struct qtreq_rename_len {
|
||||
size_t oldlen;
|
||||
size_t newlen;
|
||||
size_t flags;
|
||||
}d;
|
||||
char path[QTFS_TAIL_LEN(struct qtreq_rename_len)];
|
||||
};
|
||||
|
||||
struct qtrsp_rename {
|
||||
int ret;
|
||||
int errno;
|
||||
};
|
||||
|
||||
// xattr def
|
||||
#define QTFS_XATTR_LEN 512
|
||||
struct qtreq_xattrlist {
|
||||
size_t buffer_size;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_xattrlist {
|
||||
struct qtrsp_xattrlist_len {
|
||||
int ret;
|
||||
ssize_t size;
|
||||
}d;
|
||||
char name[QTFS_TAIL_LEN(struct qtrsp_xattrlist_len)];
|
||||
};
|
||||
|
||||
struct qtreq_xattrget {
|
||||
struct qtreq_xattrget_len {
|
||||
int pos;
|
||||
int size; // 请求最多可以读取多少字节
|
||||
char prefix_name[QTFS_XATTR_LEN];
|
||||
}d;
|
||||
char path[QTFS_TAIL_LEN(struct qtreq_xattrget_len)];
|
||||
};
|
||||
|
||||
struct qtrsp_xattrget {
|
||||
struct qtrsp_xattrget_len {
|
||||
int ret;
|
||||
int errno;
|
||||
ssize_t size;
|
||||
int pos;
|
||||
}d;
|
||||
char buf[QTFS_TAIL_LEN(struct qtrsp_xattrget_len)];
|
||||
};
|
||||
|
||||
struct qtreq_xattrset {
|
||||
struct qtreq_xattrset_len {
|
||||
int flags;
|
||||
size_t pathlen;
|
||||
size_t namelen;
|
||||
size_t valuelen;
|
||||
} d;
|
||||
/* buf: file path + name + value */
|
||||
char buf[QTFS_TAIL_LEN(struct qtreq_xattrset_len)];
|
||||
};
|
||||
|
||||
struct qtrsp_xattrset {
|
||||
int ret;
|
||||
int errno;
|
||||
};
|
||||
// xattr end
|
||||
|
||||
struct qtreq_sysmount {
|
||||
struct qtreq_sysmount_len {
|
||||
size_t dev_len;
|
||||
size_t dir_len;
|
||||
size_t type_len;
|
||||
size_t data_len;
|
||||
unsigned long flags;
|
||||
} d;
|
||||
char buf[QTFS_TAIL_LEN(struct qtreq_sysmount_len)];
|
||||
};
|
||||
|
||||
struct qtrsp_sysmount {
|
||||
int errno;
|
||||
};
|
||||
|
||||
struct qtreq_sysumount {
|
||||
int flags;
|
||||
char buf[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_sysumount {
|
||||
int errno;
|
||||
};
|
||||
|
||||
struct qtreq_poll {
|
||||
int fd;
|
||||
};
|
||||
|
||||
struct qtrsp_poll {
|
||||
int ret;
|
||||
__poll_t mask;
|
||||
};
|
||||
|
||||
|
||||
struct qtreq_epollctl {
|
||||
int fd;
|
||||
int op;
|
||||
struct qtreq_epoll_event event;
|
||||
};
|
||||
|
||||
struct qtrsp_epollctl {
|
||||
int ret;
|
||||
};
|
||||
|
||||
|
||||
// server epoll 通知 client
|
||||
#define QTFS_EPOLL_MAX_EVENTS 128
|
||||
struct qtreq_epollevt {
|
||||
unsigned int event_nums;
|
||||
struct qtreq_epoll_event events[QTFS_EPOLL_MAX_EVENTS];
|
||||
};
|
||||
#define QTFS_EPOLL_MSG_LEN (QTFS_MSG_HEAD_LEN + sizeof(struct qtreq_epollevt))
|
||||
|
||||
struct qtrsp_epollevt {
|
||||
int ret;
|
||||
};
|
||||
|
||||
struct qtreq_llseek {
|
||||
loff_t off;
|
||||
int whence;
|
||||
int fd;
|
||||
};
|
||||
|
||||
struct qtrsp_llseek {
|
||||
int ret;
|
||||
off_t off;
|
||||
};
|
||||
|
||||
struct qtreq_sc_kill {
|
||||
int pid;
|
||||
int signum;
|
||||
};
|
||||
|
||||
struct qtrsp_sc_kill {
|
||||
long ret;
|
||||
};
|
||||
|
||||
enum {
|
||||
SC_GET = 0,
|
||||
SC_SET,
|
||||
};
|
||||
#define AFFINITY_MAX_LEN (8192 / BITS_PER_LONG) // max cpu nums 8192
|
||||
struct qtreq_sc_sched_affinity {
|
||||
int type; // 0-get or 1-set
|
||||
int pid;
|
||||
size_t len;
|
||||
unsigned long user_mask_ptr[0];
|
||||
};
|
||||
|
||||
struct qtrsp_sc_sched_affinity {
|
||||
long ret;
|
||||
int len;
|
||||
unsigned long user_mask_ptr[0];
|
||||
};
|
||||
#endif
|
109
qtfs/include/symbol_wrapper.h
Normal file
109
qtfs/include/symbol_wrapper.h
Normal file
@ -0,0 +1,109 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_SYMBOL_WRAPPER_H__
|
||||
#define __QTFS_SYMBOL_WRAPPER_H__
|
||||
|
||||
#include <linux/version.h>
|
||||
typedef unsigned long (*kallsyms_lookup_name_t)(const char *name);
|
||||
extern kallsyms_lookup_name_t qtfs_kallsyms_lookup_name;
|
||||
|
||||
struct qtfs_kallsyms {
|
||||
unsigned long **sys_call_table;
|
||||
|
||||
char *(*d_absolute_path)(const struct path *, char *, int);
|
||||
#if (LINUX_VERSION_CODE <= KERNEL_VERSION(5, 10, 0))
|
||||
int (*__close_fd)(struct files_struct *, int);
|
||||
#endif
|
||||
struct task_struct *(*find_get_task_by_vpid)(pid_t nr);
|
||||
};
|
||||
|
||||
extern struct qtfs_kallsyms qtfs_kern_syms;
|
||||
int qtfs_kallsyms_hack_init(void);
|
||||
|
||||
#ifdef QTFS_CLIENT
|
||||
enum {
|
||||
SYMBOL_SYSCALL_MOUNT,
|
||||
SYMBOL_SYSCALL_UMOUNT,
|
||||
SYMBOL_SYSCALL_EPOLL_CTL,
|
||||
SYMBOL_SYSCALL_CONNECT,
|
||||
SYMBOL_MAX_NUM,
|
||||
};
|
||||
#endif
|
||||
#ifdef QTFS_SERVER
|
||||
enum {
|
||||
SYMBOL_SYSCALL_CONNECT,
|
||||
SYMBOL_MAX_NUM,
|
||||
};
|
||||
#endif
|
||||
extern unsigned long *symbols_origin[SYMBOL_MAX_NUM];
|
||||
|
||||
#ifdef __x86_64__
|
||||
// make the page writeable
|
||||
static inline int make_rw(unsigned long address)
|
||||
{
|
||||
unsigned int level;
|
||||
pte_t *pte = lookup_address(address, &level);
|
||||
pte->pte |= _PAGE_RW;
|
||||
return 0;
|
||||
}
|
||||
|
||||
// make the page write protected
|
||||
static inline int make_ro(unsigned long address)
|
||||
{
|
||||
unsigned int level;
|
||||
pte_t *pte = lookup_address(address, &level);
|
||||
pte->pte &= ~_PAGE_RW;
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef __aarch64__
|
||||
extern void (*update_mapping_prot)(phys_addr_t phys, unsigned long virt, phys_addr_t size, pgprot_t prot);
|
||||
extern unsigned long start_rodata, end_rodata;
|
||||
#define section_size (end_rodata - start_rodata)
|
||||
#endif
|
||||
|
||||
int qtfs_syscall_replace_start(void);
|
||||
void qtfs_syscall_replace_stop(void);
|
||||
|
||||
noinline long qtfs_syscall_umount(char __user *name, int flags);
|
||||
noinline long qtfs_syscall_mount(char __user *dev_name, char __user *dir_name,
|
||||
char __user *type, unsigned long flags, void __user *data);
|
||||
noinline long qtfs_syscall_epoll_ctl(int epfd, int op, int fd, struct epoll_event __user *event);
|
||||
noinline long qtfs_syscall_unlink(const char __user *pathname);
|
||||
noinline int qtfs_syscall_readlinkat(int dfd, const char __user *path,
|
||||
char __user *buf, int bufsiz);
|
||||
noinline int qtfs_syscall_renameat2(int olddfd, const char __user *oldname,
|
||||
int newdfd, const char __user *newname, unsigned int flags);
|
||||
noinline long qtfs_syscall_mkdirat(int dfd, const char __user *pathname, umode_t mode);
|
||||
noinline long qtfs_syscall_rmdir(const char __user *pathname);
|
||||
noinline int qtfs_syscall_statfs(const char __user *path, struct statfs __user *buf);
|
||||
noinline int qtfs_syscall_openat(int dfd, const char __user *filename, int flags,
|
||||
umode_t mode);
|
||||
noinline int qtfs_syscall_epoll_wait(int epfd, struct epoll_event __user *events,
|
||||
int maxevents, int timout);
|
||||
noinline int qtfs_syscall_linkat(int olddfd, const char __user *oldname,
|
||||
int newdfd, const char __user *newname, int flags);
|
||||
noinline long qtfs_syscall_mknodat(int dfd, const char __user *filename, umode_t mode,
|
||||
unsigned int dev);
|
||||
noinline off_t qtfs_syscall_lseek(unsigned int fd, off_t offset, unsigned int whence);
|
||||
long qtfs_syscall_write(unsigned int fd, const char __user *buf, size_t count);
|
||||
long qtfs_syscall_read(unsigned int fd, char __user *buf, size_t count);
|
||||
long qtfs_syscall_ioctl(unsigned int fd, unsigned int cmd, unsigned long arg);
|
||||
long qtfs_syscall_kill(pid_t pid, int sig);
|
||||
long qtfs_syscall_sched_getaffinity(pid_t pid, unsigned int len, unsigned long __user *user_mask_ptr);
|
||||
long qtfs_syscall_sched_setaffinity(pid_t pid, unsigned int len, unsigned long __user *user_mask_ptr);
|
||||
long qtfs_syscall_connect(int fd, struct sockaddr __user *uservaddr, int addrlen);
|
||||
#endif
|
||||
|
38
qtfs/ipc/Makefile
Normal file
38
qtfs/ipc/Makefile
Normal file
@ -0,0 +1,38 @@
|
||||
DEPGLIB=-lglib-2.0 -I../ -I../include/ -I/usr/include/glib-2.0 -I/usr/lib64/glib-2.0/include -lpthread
|
||||
CFLAGS += -g -O2
|
||||
CFLAGS += -fstack-protector-strong
|
||||
CFLAGS += -fPIE -pie -fPIC
|
||||
CFLAGS += -D_FORTIFY_SOURCE=2
|
||||
ifdef UDS_TEST_MODE
|
||||
CFLAGS += -DUDS_TEST_MODE
|
||||
endif
|
||||
LDFLAGS += -Wl,-z,now
|
||||
LDFLAGS += -Wl,-z,noexecstack
|
||||
LDFLAGS += -fPIE -pie
|
||||
|
||||
all: udsproxyd libudsproxy.so
|
||||
|
||||
udsproxyd: uds_event.o uds_main.o
|
||||
gcc $(LDFLAGS) -o udsproxyd $^ -I../ $(DEPGLIB)
|
||||
@test -z $(UDS_TEST_MODE) || echo "Important risk warning: The test mode is turned on,\
|
||||
and udsproxyd will expose the network port, which will bring security risks and is only for\
|
||||
testing! If you do not understand the risks, please don't use or compile again without\
|
||||
UDS_TEST_MODE."
|
||||
|
||||
uds_event.o:
|
||||
cc $(CFLAGS) -c -o uds_event.o uds_event.c $(DEPGLIB)
|
||||
|
||||
uds_main.o:
|
||||
cc $(CFLAGS) -c -o uds_main.o uds_main.c $(DEPGLIB)
|
||||
|
||||
libudsproxy.so:
|
||||
gcc $(CFLAGS) $(LDFLAGS) -o libudsproxy.so uds_connector.c --shared
|
||||
|
||||
install:
|
||||
yes | cp udsproxyd /usr/bin/
|
||||
yes | cp libudsproxy.so /usr/lib64/
|
||||
|
||||
clean:
|
||||
@rm -rf *.o udsproxyd libudsproxy.so
|
||||
|
||||
.PHONY: clean
|
160
qtfs/ipc/uds_connector.c
Normal file
160
qtfs/ipc/uds_connector.c
Normal file
@ -0,0 +1,160 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
#include <unistd.h>
|
||||
#include <signal.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/un.h>
|
||||
#include <time.h>
|
||||
#include <dlfcn.h>
|
||||
#include <sys/types.h>
|
||||
|
||||
#include "uds_module.h"
|
||||
|
||||
#define LOG_OFF 1
|
||||
#if LOG_OFF
|
||||
#define uds_log(info, ...)
|
||||
#define uds_err(info, ...)
|
||||
#else
|
||||
#define uds_log(info, ...) \
|
||||
do { \
|
||||
time_t t; \
|
||||
struct tm p; \
|
||||
time(&t); \
|
||||
localtime_r(&t, &p); \
|
||||
printf("[%d/%02d/%02d %02d:%02d:%02d][LOG:%s:%3d]"info"\n", \
|
||||
p.tm_year + 1900, p.tm_mon+1, p.tm_mday, \
|
||||
p.tm_hour, p.tm_min, p.tm_sec, __func__, __LINE__, ##__VA_ARGS__); \
|
||||
} while (0);
|
||||
|
||||
#define uds_err(info, ...) \
|
||||
do { \
|
||||
time_t t; \
|
||||
struct tm p; \
|
||||
time(&t); \
|
||||
localtime_r(&t, &p); \
|
||||
printf("[%d/%02d/%02d %02d:%02d:%02d][LOG:%s:%3d]"info"\n", \
|
||||
p.tm_year + 1900, p.tm_mon+1, p.tm_mday, \
|
||||
p.tm_hour, p.tm_min, p.tm_sec, __func__, __LINE__, ##__VA_ARGS__); \
|
||||
} while (0);
|
||||
#endif
|
||||
|
||||
static unsigned short uds_conn_get_sock_type(int sockfd)
|
||||
{
|
||||
unsigned short type;
|
||||
socklen_t len = 2;
|
||||
int ret = getsockopt(sockfd, SOL_SOCKET, SO_TYPE, &type, &len);
|
||||
if (ret < 0) {
|
||||
uds_err("get sock type failed, fd:%d", sockfd);
|
||||
return (unsigned short)-1;
|
||||
}
|
||||
uds_log("fd:%d type:%d", sockfd, type);
|
||||
return type;
|
||||
}
|
||||
|
||||
static int uds_conn_whitelist_check(const char *path)
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
|
||||
int connect(int fd, const struct sockaddr *addrarg, socklen_t len)
|
||||
{
|
||||
int sock_fd;
|
||||
typeof(connect) *libcconnect = NULL;
|
||||
int libcret;
|
||||
const struct sockaddr_un *addr = (const struct sockaddr_un *)addrarg;
|
||||
|
||||
if (libcconnect == NULL) {
|
||||
libcconnect = dlsym(((void *) - 1l), "connect");
|
||||
if (libcconnect == NULL) {
|
||||
uds_err("can't find connect by dlsym.");
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
libcret = (*libcconnect)(fd, addrarg, len);
|
||||
if (libcret == 0 || addr->sun_family != AF_UNIX) {
|
||||
// 如果本地connect成功,或者非UNIX DOMAIN SOCKET,都直接返回即可
|
||||
return libcret;
|
||||
}
|
||||
|
||||
if (strlen(addr->sun_path) >= (UDS_SUN_PATH_LEN - strlen(UDS_PROXY_SUFFIX))) {
|
||||
uds_err("sun_path:<%s> len:%d is too large to add suffex:<%s>, so can't connect to uds proxy.",
|
||||
addr->sun_path, strlen(addr->sun_path), UDS_PROXY_SUFFIX);
|
||||
return libcret;
|
||||
}
|
||||
|
||||
uds_log("enter uds connect fd:%d sunpath:%s family:%d len:%d ", fd, addr->sun_path, addr->sun_family, len);
|
||||
// 本地未连接,且是uds链接
|
||||
if (!uds_conn_whitelist_check(addr->sun_path)) {
|
||||
uds_err("path:%s not in white list", addr->sun_path);
|
||||
return libcret;
|
||||
}
|
||||
|
||||
// 尝试远端链接
|
||||
do {
|
||||
int ret;
|
||||
struct uds_proxy_remote_conn_req remoteconn;
|
||||
struct uds_proxy_remote_conn_rsp remotersp;
|
||||
struct sockaddr_un proxy = {.sun_family = AF_UNIX};
|
||||
sock_fd = socket(AF_UNIX, SOCK_STREAM, 0);
|
||||
if (sock_fd < 0) {
|
||||
uds_err("create socket failed");
|
||||
return libcret;
|
||||
}
|
||||
|
||||
strncpy(proxy.sun_path, UDS_BUILD_CONN_ADDR, sizeof(proxy.sun_path));
|
||||
if ((*libcconnect)(sock_fd, (struct sockaddr *)&proxy, sizeof(struct sockaddr_un)) < 0) {
|
||||
uds_err("can't connect to uds proxy: %s", UDS_BUILD_CONN_ADDR);
|
||||
goto err_end;
|
||||
}
|
||||
// 这里type需要是第一个入参fd的type
|
||||
remoteconn.type = uds_conn_get_sock_type(fd);
|
||||
if (remoteconn.type == (unsigned short)-1) {
|
||||
remoteconn.type = SOCK_STREAM;
|
||||
}
|
||||
memset(remoteconn.sun_path, 0, sizeof(remoteconn.sun_path));
|
||||
strncpy(remoteconn.sun_path, addr->sun_path, sizeof(remoteconn.sun_path));
|
||||
ret = send(sock_fd, &remoteconn, sizeof(remoteconn), 0);
|
||||
if (ret <= 0) {
|
||||
uds_err("send remote connect request failed, ret:%d errno:%d", ret, errno);
|
||||
goto err_end;
|
||||
}
|
||||
ret = recv(sock_fd, &remotersp, sizeof(remotersp), MSG_WAITALL);
|
||||
if (ret <= 0) {
|
||||
uds_err("recv remote connect replay failed, ret:%d errno:%d", ret, errno);
|
||||
goto err_end;
|
||||
}
|
||||
if (remotersp.ret == 0) {
|
||||
goto err_end;
|
||||
}
|
||||
} while(0);
|
||||
|
||||
close(sock_fd);
|
||||
|
||||
struct sockaddr_un addr_proxy;
|
||||
int sun_len = strlen(addr->sun_path);
|
||||
memcpy(&addr_proxy, addr, sizeof(struct sockaddr_un));
|
||||
memcpy(&addr_proxy.sun_path[sun_len], UDS_PROXY_SUFFIX, strlen(UDS_PROXY_SUFFIX));
|
||||
addr_proxy.sun_path[sun_len + strlen(UDS_PROXY_SUFFIX)] = '\0';
|
||||
return (*libcconnect)(fd, (const struct sockaddr *)&addr_proxy, sizeof(struct sockaddr_un));
|
||||
|
||||
err_end:
|
||||
close(sock_fd);
|
||||
return libcret;
|
||||
}
|
1096
qtfs/ipc/uds_event.c
Normal file
1096
qtfs/ipc/uds_event.c
Normal file
File diff suppressed because it is too large
Load Diff
81
qtfs/ipc/uds_event.h
Normal file
81
qtfs/ipc/uds_event.h
Normal file
@ -0,0 +1,81 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#ifndef __QTFS_UDS_EVENT_H__
|
||||
#define __QTFS_UDS_EVENT_H__
|
||||
|
||||
#define UDS_EVENT_BUFLEN 4096
|
||||
#define UDS_PATH_MAX 1024
|
||||
|
||||
#define UDS_EVENT_WAIT_TMOUT 5 // 5s timeout
|
||||
|
||||
enum EVENT_RETCODE {
|
||||
EVENT_OK = 0,
|
||||
EVENT_ERR = -1,
|
||||
EVENT_DEL = -2, // del this event after return
|
||||
};
|
||||
|
||||
enum TCP2TCP_TYPE {
|
||||
MSG_NORMAL = 0xa5a5, // 消息类型从特殊数字开始,防止误识别消息
|
||||
MSG_SCM_RIGHTS,
|
||||
MSG_SCM_CREDENTIALS, // unix domain 扩展消息,预留
|
||||
MSG_SCM_SECURITY, // unix domain 扩展消息,预留
|
||||
MSG_GET_TARGET, // 控制消息,用于获取对端的target fd
|
||||
MSG_SCM_PIPE, // 使用SCM传递了一个pipe
|
||||
MSG_END, // tcp消息的结束体
|
||||
};
|
||||
|
||||
enum TCPCNTL_TYPE {
|
||||
MSGCNTL_UDS = 1, // uds代理模式
|
||||
MSGCNTL_PIPE, // pipe匿名管道代理模式
|
||||
};
|
||||
|
||||
// 因为要区分SCM_RIGHTS和普通消息,TCP到TCP需要有一个协议头
|
||||
struct uds_tcp2tcp {
|
||||
int msgtype;
|
||||
int msglen; // len of data
|
||||
char data[0];
|
||||
};
|
||||
|
||||
struct uds_msg_scmrights {
|
||||
int flags; // open flags
|
||||
char path[UDS_PATH_MAX];
|
||||
};
|
||||
|
||||
enum {
|
||||
SCM_PIPE_READ = 0,
|
||||
SCM_PIPE_WRITE,
|
||||
SCM_PIPE_NUM,
|
||||
};
|
||||
|
||||
struct uds_stru_scm_pipe {
|
||||
int dir; // 0: send read filedes; 1: send write filedes
|
||||
// proxy通过scm rights接收到员pipe fd,后面消息回来时事件
|
||||
// 会发生变化,所以需要回消息时带上,才能建立关联
|
||||
int srcfd;
|
||||
};
|
||||
|
||||
int uds_event_uds_listener(void *arg, int epfd, struct uds_event_global_var *p_event_var);
|
||||
int uds_event_tcp_listener(void *arg, int epfd, struct uds_event_global_var *p_event_var);
|
||||
int uds_event_diag_info(void *arg, int epfd, struct uds_event_global_var *p_event_var);
|
||||
int uds_event_debug_level(void *arg, int epfd, struct uds_event_global_var *p_event_var);
|
||||
int uds_event_pre_handler(struct uds_event *evt);
|
||||
int uds_event_pre_hook(struct uds_event_global_var *p_event_var);
|
||||
int uds_event_post_hook(struct uds_event_global_var *p_event_var);
|
||||
int uds_event_module_init(struct uds_event_global_var *p_event_var);
|
||||
void uds_event_module_fini(struct uds_event_global_var *p);
|
||||
|
||||
#endif
|
||||
|
758
qtfs/ipc/uds_main.c
Normal file
758
qtfs/ipc/uds_main.c
Normal file
@ -0,0 +1,758 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <pthread.h>
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
#include <unistd.h>
|
||||
#include <signal.h>
|
||||
#include <sys/epoll.h>
|
||||
#include <netinet/ip.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/un.h>
|
||||
#include <netinet/udp.h>
|
||||
#include <netinet/tcp.h>
|
||||
#include <sys/socket.h>
|
||||
#include <arpa/inet.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <dirent.h>
|
||||
#include <glib.h>
|
||||
#include <sys/resource.h>
|
||||
#include <sys/prctl.h>
|
||||
#include <sys/file.h>
|
||||
#include <linux/vm_sockets.h>
|
||||
|
||||
#include "comm.h"
|
||||
#include "uds_main.h"
|
||||
#include "uds_event.h"
|
||||
|
||||
char *uds_log_str[UDS_LOG_MAX + 1] = {"NONE", "ERROR", "INFO", "UNKNOWN"};
|
||||
struct uds_global_var g_uds_var;
|
||||
struct uds_global_var *p_uds_var = &g_uds_var;
|
||||
struct uds_event_global_var *g_event_var = NULL;
|
||||
GHashTable *event_tmout_hash;
|
||||
|
||||
struct uds_event *uds_alloc_event()
|
||||
{
|
||||
struct uds_event *p = (struct uds_event *)malloc(sizeof(struct uds_event));
|
||||
if (p == NULL) {
|
||||
uds_err("malloc failed.");
|
||||
return NULL;
|
||||
}
|
||||
memset(p, 0, sizeof(struct uds_event));
|
||||
return p;
|
||||
}
|
||||
|
||||
int uds_event_insert(int efd, struct uds_event *event)
|
||||
{
|
||||
struct epoll_event evt;
|
||||
evt.data.ptr = (void *)event;
|
||||
evt.events = EPOLLIN;
|
||||
if (-1 == epoll_ctl(efd, EPOLL_CTL_ADD, event->fd, &evt)) {
|
||||
uds_err("epoll ctl add fd:%d event failed.", event->fd);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int uds_event_suspend(int efd, struct uds_event *event)
|
||||
{
|
||||
int ret = epoll_ctl(efd, EPOLL_CTL_DEL, event->fd, NULL);
|
||||
if (ret != 0) {
|
||||
uds_err("failed to suspend fd:%d.", event->fd);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int uds_event_delete(int efd, int fd)
|
||||
{
|
||||
close(fd);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int uds_recv_with_timeout(int fd, char *msg, int len)
|
||||
{
|
||||
#define TMOUT_BLOCK_SIZE 1024
|
||||
#define TMOUT_UNIT_MS 20
|
||||
#define TMOUT_INTERVAL 1
|
||||
#define TMOUT_MAX_MS 1000
|
||||
int total_recv = 0;
|
||||
int ret;
|
||||
int tmout_ms = ((len / TMOUT_BLOCK_SIZE) + 1) * TMOUT_UNIT_MS;
|
||||
if (len <= 0 || msg == NULL || fd < 0) {
|
||||
uds_err("invalid param fd:%d len:%d or %s", fd, len, (msg == NULL) ? "msg is NULL" : "msg is not NULL");
|
||||
return 0;
|
||||
}
|
||||
if (tmout_ms > TMOUT_MAX_MS)
|
||||
tmout_ms = TMOUT_MAX_MS;
|
||||
do {
|
||||
ret = recv(fd, &msg[total_recv], len - total_recv, 0);
|
||||
if (ret < 0) {
|
||||
uds_err("recv failed ret:%d errno:%d", ret, errno);
|
||||
return ret;
|
||||
}
|
||||
total_recv += ret;
|
||||
if (total_recv > len) {
|
||||
uds_err("fatal error total recv:%d longger than target len:%d", total_recv, len);
|
||||
return 0;
|
||||
}
|
||||
if (total_recv == len) {
|
||||
return total_recv;
|
||||
}
|
||||
usleep(TMOUT_INTERVAL * 1000);
|
||||
tmout_ms -= TMOUT_INTERVAL;
|
||||
} while (tmout_ms > 0);
|
||||
uds_err("Fatal error, the target recv len:%d and only %d length is received when it time out", len, total_recv);
|
||||
return 0;
|
||||
}
|
||||
|
||||
#pragma GCC diagnostic ignored "-Wpointer-to-int-cast"
|
||||
int uds_event_tmout_item(gpointer key, gpointer value, gpointer data)
|
||||
{
|
||||
struct uds_event *evt = (struct uds_event *)value;
|
||||
if (evt->tmout == 0) {
|
||||
uds_err("Unexpected time out value in fd:%d", key);
|
||||
goto clear;
|
||||
}
|
||||
evt->tmout--;
|
||||
if (evt->tmout > 0)
|
||||
return 0;
|
||||
|
||||
uds_log("The connection was not established within 5s, the fd:%d wait was over due to a timeout.", key);
|
||||
clear:
|
||||
close((int)key);
|
||||
free(evt);
|
||||
return 1;
|
||||
}
|
||||
#pragma GCC diagnostic pop
|
||||
|
||||
void uds_event_timeout_proc()
|
||||
{
|
||||
g_hash_table_foreach_remove(event_tmout_hash, uds_event_tmout_item, NULL);
|
||||
}
|
||||
|
||||
void uds_main_loop(int efd, struct uds_thread_arg *arg)
|
||||
{
|
||||
int n = 0;
|
||||
int ret;
|
||||
struct uds_event *udsevt;
|
||||
struct epoll_event *evts = NULL;
|
||||
struct uds_event_global_var *p_event_var = arg->p_event_var;
|
||||
if (p_event_var == NULL) {
|
||||
uds_err("event variable invalid.");
|
||||
return;
|
||||
}
|
||||
|
||||
evts = calloc(UDS_EPOLL_MAX_EVENTS, sizeof(struct epoll_event));
|
||||
if (evts == NULL) {
|
||||
uds_err("init calloc evts failed.");
|
||||
return;
|
||||
}
|
||||
if (uds_event_module_init(p_event_var) == EVENT_ERR) {
|
||||
uds_err("uds event module init failed, main loop not run.");
|
||||
free(evts);
|
||||
return;
|
||||
}
|
||||
#ifdef QTFS_SERVER
|
||||
extern int engine_run;
|
||||
while (engine_run) {
|
||||
#else
|
||||
while (1) {
|
||||
#endif
|
||||
n = epoll_wait(efd, evts, UDS_EPOLL_MAX_EVENTS, 1000);
|
||||
if (n == 0) {
|
||||
uds_event_timeout_proc();
|
||||
continue;
|
||||
}
|
||||
if (n < 0) {
|
||||
uds_err("epoll wait return errcode:%d", n);
|
||||
continue;
|
||||
}
|
||||
arg->info.events += n;
|
||||
uds_event_pre_hook(p_event_var);
|
||||
for (int i = 0; i < n; i++) {
|
||||
udsevt = (struct uds_event *)evts[i].data.ptr;
|
||||
uds_log("event fd:%d events:%d tofree:%u", udsevt->fd, evts[i].events, udsevt->tofree);
|
||||
if (udsevt->handler == NULL) {
|
||||
uds_err("bad event, fd:%d handler is NULL.", udsevt->fd);
|
||||
continue;
|
||||
}
|
||||
// 预检查失败择不执行handler
|
||||
if (uds_event_pre_handler(udsevt) == EVENT_ERR) {
|
||||
continue;
|
||||
}
|
||||
ret = udsevt->handler(udsevt, efd, p_event_var);
|
||||
// 此处释放当前事件,peer事件需要handler里面释放
|
||||
if (ret == EVENT_DEL) {
|
||||
uds_del_event(udsevt);
|
||||
}
|
||||
}
|
||||
uds_event_post_hook(p_event_var);
|
||||
}
|
||||
free(evts);
|
||||
uds_log("main loop exit.");
|
||||
uds_event_module_fini(p_event_var);
|
||||
return;
|
||||
}
|
||||
|
||||
#define UDS_MAX_LISTEN_NUM 64
|
||||
int uds_build_tcp_connection(struct uds_conn_arg *arg)
|
||||
{
|
||||
int family = AF_VSOCK;
|
||||
if (arg->cs > UDS_SOCKET_SERVER) {
|
||||
uds_err("cs type %d is error.", arg->cs);
|
||||
return -1;
|
||||
}
|
||||
#ifdef UDS_TEST_MODE
|
||||
family = AF_INET;
|
||||
struct sockaddr_in sock_addr;
|
||||
memset(&sock_addr, 0, sizeof(sock_addr));
|
||||
sock_addr.sin_family = AF_INET;
|
||||
#else
|
||||
family = AF_VSOCK;
|
||||
struct sockaddr_vm sock_addr;
|
||||
memset(&sock_addr, 0, sizeof(sock_addr));
|
||||
sock_addr.svm_family = AF_VSOCK;
|
||||
#endif
|
||||
|
||||
int sock_fd = socket(family, SOCK_STREAM, 0);
|
||||
if (sock_fd < 0) {
|
||||
uds_err("As %s failed, socket fd: %d, errno:%d.",
|
||||
(arg->cs == UDS_SOCKET_CLIENT) ? "client" : "server",
|
||||
sock_fd, errno);
|
||||
return -1;
|
||||
}
|
||||
arg->sockfd = sock_fd;
|
||||
|
||||
if (arg->cs == UDS_SOCKET_SERVER) {
|
||||
#ifdef UDS_TEST_MODE
|
||||
sock_addr.sin_port = htons(p_uds_var->tcp.port);
|
||||
sock_addr.sin_addr.s_addr = inet_addr(p_uds_var->tcp.addr);
|
||||
#else
|
||||
sock_addr.svm_port = p_uds_var->vsock.port;
|
||||
sock_addr.svm_cid = p_uds_var->vsock.cid;
|
||||
#endif
|
||||
if (bind(sock_fd, (struct sockaddr *)&sock_addr, sizeof(sock_addr)) < 0) {
|
||||
uds_err("As tcp server failed, bind error, errno:%d.",
|
||||
errno);
|
||||
goto close_and_return;
|
||||
}
|
||||
if (listen(sock_fd, UDS_MAX_LISTEN_NUM) < 0) {
|
||||
uds_err("As tcp server listen failed, errno:%d.", errno);
|
||||
goto close_and_return;
|
||||
}
|
||||
} else {
|
||||
#ifdef UDS_TEST_MODE
|
||||
sock_addr.sin_port = htons(p_uds_var->tcp.peerport);
|
||||
sock_addr.sin_addr.s_addr = inet_addr(p_uds_var->tcp.peeraddr);
|
||||
#else
|
||||
sock_addr.svm_port = p_uds_var->vsock.peerport;
|
||||
sock_addr.svm_cid = p_uds_var->vsock.peercid;
|
||||
#endif
|
||||
if (connect(arg->sockfd, (struct sockaddr *)&sock_addr, sizeof(sock_addr)) < 0) {
|
||||
goto close_and_return;
|
||||
}
|
||||
arg->connfd = sock_fd;
|
||||
#ifdef UDS_TEST_MODE
|
||||
uds_log("Connect to tcp server successed, ip:%s port:%u", p_uds_var->tcp.peeraddr, p_uds_var->tcp.peerport);
|
||||
#else
|
||||
uds_log("Connect to vsock server successed, cid:%u port:%u", p_uds_var->vsock.peercid, p_uds_var->vsock.peerport);
|
||||
#endif
|
||||
}
|
||||
|
||||
return 0;
|
||||
close_and_return:
|
||||
close(sock_fd);
|
||||
return -1;
|
||||
}
|
||||
|
||||
int uds_build_unix_connection(struct uds_conn_arg *arg)
|
||||
{
|
||||
if (arg->cs > UDS_SOCKET_SERVER) {
|
||||
uds_err("cs type %d is error.", arg->cs);
|
||||
return -1;
|
||||
}
|
||||
struct sockaddr_un sock_addr = {
|
||||
.sun_family = AF_UNIX,
|
||||
};
|
||||
int sock_fd = socket(AF_UNIX, arg->udstype, 0);
|
||||
|
||||
if (sock_fd < 0) {
|
||||
uds_err("As %s failed, socket fd: %d, errno:%d.",
|
||||
(arg->cs == UDS_SOCKET_CLIENT) ? "client" : "server",
|
||||
sock_fd, errno);
|
||||
return -1;
|
||||
}
|
||||
strncpy(sock_addr.sun_path, arg->sun_path, sizeof(sock_addr.sun_path));
|
||||
arg->sockfd = sock_fd;
|
||||
|
||||
if (arg->cs == UDS_SOCKET_SERVER) {
|
||||
unlink(sock_addr.sun_path);
|
||||
if (bind(sock_fd, (struct sockaddr *)&sock_addr, sizeof(sock_addr)) < 0) {
|
||||
uds_err("As uds server failed, bind error, errno:%d.",
|
||||
errno);
|
||||
goto close_and_return;
|
||||
}
|
||||
if (listen(sock_fd, UDS_MAX_LISTEN_NUM) < 0) {
|
||||
uds_err("As uds server listen failed, errno:%d.", errno);
|
||||
goto close_and_return;
|
||||
}
|
||||
} else {
|
||||
if (connect(arg->sockfd, (struct sockaddr *)&sock_addr, sizeof(struct sockaddr_un)) < 0) {
|
||||
goto close_and_return;
|
||||
}
|
||||
arg->connfd = sock_fd;
|
||||
uds_log("Connect to uds server successed, sun path:%s", arg->sun_path);
|
||||
}
|
||||
|
||||
return 0;
|
||||
close_and_return:
|
||||
uds_log("close sockfd:%d and return", sock_fd);
|
||||
close(sock_fd);
|
||||
return -1;
|
||||
|
||||
}
|
||||
|
||||
int uds_sock_step_accept(int sock_fd, int family)
|
||||
{
|
||||
struct sockaddr_in in_addr;
|
||||
struct sockaddr_un un_addr;
|
||||
socklen_t len = (family == AF_INET) ? sizeof(struct sockaddr_in) : sizeof(struct sockaddr_un);
|
||||
int connfd;
|
||||
if (family == AF_INET) {
|
||||
connfd = accept(sock_fd, (struct sockaddr *)&in_addr, &len);
|
||||
} else {
|
||||
connfd = accept(sock_fd, (struct sockaddr *)&un_addr, &len);
|
||||
}
|
||||
if (connfd < 0) {
|
||||
uds_err("Accept error:%d, errno:%d.", connfd, errno);
|
||||
return connfd;
|
||||
}
|
||||
if (family == AF_INET) {
|
||||
uds_log("Accept success, ip:%s, port:%u",
|
||||
inet_ntoa(in_addr.sin_addr),
|
||||
ntohs(in_addr.sin_port));
|
||||
} else {
|
||||
uds_log("Accept success, sun path:%s", un_addr.sun_path);
|
||||
}
|
||||
return connfd;
|
||||
}
|
||||
|
||||
struct uds_event *uds_add_event(int fd, struct uds_event *peer, int (*handler)(void *, int, struct uds_event_global_var *), void *priv)
|
||||
{
|
||||
struct uds_event *newevt = uds_alloc_event();
|
||||
int hash = fd % p_uds_var->work_thread_num;
|
||||
if (newevt == NULL || p_uds_var->efd[hash] <= 0) {
|
||||
uds_err("alloc event failed, efd:%d hash:%d", p_uds_var->efd[hash], hash);
|
||||
if(newevt != NULL) {
|
||||
free(newevt);
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
newevt->fd = fd;
|
||||
newevt->peer = peer; // 如果tcp回应,消息转回uds这个fd
|
||||
newevt->handler = handler;
|
||||
newevt->priv = priv;
|
||||
newevt->tofree = 0;
|
||||
newevt->tmout = 0;
|
||||
if (uds_event_insert(p_uds_var->efd[hash], newevt) != 0) {
|
||||
uds_del_event(newevt);
|
||||
return NULL;
|
||||
}
|
||||
return newevt;
|
||||
}
|
||||
|
||||
struct uds_event *uds_add_pipe_event(int fd, int peerfd, int (*handler)(void *, int, struct uds_event_global_var *), void *priv)
|
||||
{
|
||||
int hash = fd % p_uds_var->work_thread_num;
|
||||
struct uds_event *newevt = uds_alloc_event();
|
||||
if (newevt == NULL || p_uds_var->efd[hash] <= 0) {
|
||||
uds_err("alloc event failed, efd:%d", p_uds_var->efd[hash]);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
newevt->fd = fd;
|
||||
newevt->peerfd = peerfd; // 如果tcp回应,消息转回uds这个fd
|
||||
newevt->handler = handler;
|
||||
newevt->priv = priv;
|
||||
newevt->tofree = 0;
|
||||
newevt->pipe = 1;
|
||||
newevt->tmout = 0;
|
||||
if (uds_event_insert(p_uds_var->efd[hash], newevt) != 0) {
|
||||
uds_del_event(newevt);
|
||||
return NULL;
|
||||
}
|
||||
return newevt;
|
||||
}
|
||||
|
||||
void uds_del_event(struct uds_event *evt)
|
||||
{
|
||||
int hash = evt->fd % p_uds_var->work_thread_num;
|
||||
if (evt->pipe == 1 && evt->peerfd != -1) {
|
||||
// pipe是单向,peerfd没有epoll事件,所以直接关闭
|
||||
close(evt->peerfd);
|
||||
evt->peerfd = -1;
|
||||
}
|
||||
uds_event_delete(p_uds_var->efd[hash], evt->fd);
|
||||
free(evt);
|
||||
return;
|
||||
}
|
||||
|
||||
void uds_thread_diag_init(struct uds_thread_info *info)
|
||||
{
|
||||
info->events = 0;
|
||||
info->fdnum = 0;
|
||||
}
|
||||
|
||||
void *uds_proxy_thread(void *arg)
|
||||
{
|
||||
struct uds_thread_arg *parg = (struct uds_thread_arg *)arg;
|
||||
// set thread name to "udsproxyd"
|
||||
prctl(PR_SET_NAME, (unsigned long)"udsproxyd");
|
||||
uds_thread_diag_init(&parg->info);
|
||||
uds_main_loop(parg->efd, parg);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
struct uds_event *uds_init_unix_listener(const char *addr, int (*handler)(void *, int, struct uds_event_global_var *))
|
||||
{
|
||||
struct uds_event *udsevt;
|
||||
struct uds_conn_arg arg;
|
||||
struct uds_conn_arg *parg = &arg;
|
||||
|
||||
parg->cs = UDS_SOCKET_SERVER;
|
||||
strncpy(parg->sun_path, addr, sizeof(parg->sun_path) - 1);
|
||||
parg->udstype = SOCK_STREAM;
|
||||
if (uds_build_unix_connection(parg) != 0)
|
||||
return NULL;
|
||||
udsevt = uds_add_event(parg->sockfd, NULL, handler, NULL);
|
||||
if (udsevt == NULL) {
|
||||
close(parg->sockfd);
|
||||
uds_err("add unix listener event failed.");
|
||||
return NULL;
|
||||
}
|
||||
return udsevt;
|
||||
}
|
||||
|
||||
struct uds_event *uds_init_tcp_listener()
|
||||
{
|
||||
struct uds_event *tcpevt;
|
||||
struct uds_conn_arg arg;
|
||||
struct uds_conn_arg *parg = &arg;
|
||||
parg->cs = UDS_SOCKET_SERVER;
|
||||
if (uds_build_tcp_connection(parg) != 0)
|
||||
return NULL;
|
||||
|
||||
tcpevt = uds_add_event(parg->sockfd, NULL, uds_event_tcp_listener, NULL);
|
||||
if (tcpevt == NULL)
|
||||
return NULL;
|
||||
return tcpevt;
|
||||
}
|
||||
|
||||
static inline void uds_rollback_efd(int *efd, int maxidx)
|
||||
{
|
||||
for (int i = 0; i < maxidx; i++) {
|
||||
close(efd[i]);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
static int uds_thread_execute()
|
||||
{
|
||||
pthread_t *thrd = (pthread_t *)malloc(sizeof(pthread_t) * p_uds_var->work_thread_num);
|
||||
if (thrd == NULL) {
|
||||
uds_err("thread info malloc failed.");
|
||||
return -1;
|
||||
}
|
||||
|
||||
for (int i = 0; i < p_uds_var->work_thread_num; i++) {
|
||||
p_uds_var->work_thread[i].p_event_var = &g_event_var[i];
|
||||
p_uds_var->work_thread[i].efd = p_uds_var->efd[i];
|
||||
(void)pthread_create(&thrd[i], NULL, uds_proxy_thread, &p_uds_var->work_thread[i]);
|
||||
}
|
||||
p_uds_var->loglevel = UDS_LOG_NONE;
|
||||
for (int i = 0; i < p_uds_var->work_thread_num; i++)
|
||||
pthread_join(thrd[i], NULL);
|
||||
free(thrd);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void uds_thread_create()
|
||||
{
|
||||
struct uds_conn_arg arg;
|
||||
struct uds_conn_arg *parg = &arg;
|
||||
struct uds_event *udsevt;
|
||||
struct uds_event *tcpevt;
|
||||
struct uds_event *diagevt;
|
||||
struct uds_event *logevt;
|
||||
int efd;
|
||||
int ret;
|
||||
|
||||
for (int i = 0; i < p_uds_var->work_thread_num; i++) {
|
||||
efd = epoll_create1(0);
|
||||
if (efd == -1) {
|
||||
uds_rollback_efd(p_uds_var->efd, i);
|
||||
uds_err("epoll create1 failed, i:%d.", i);
|
||||
return;
|
||||
}
|
||||
p_uds_var->efd[i] = efd;
|
||||
}
|
||||
|
||||
if ((udsevt = uds_init_unix_listener(UDS_BUILD_CONN_ADDR, uds_event_uds_listener)) == NULL)
|
||||
goto rollbackefd;
|
||||
|
||||
if ((tcpevt = uds_init_tcp_listener()) == NULL)
|
||||
goto end;
|
||||
|
||||
if ((diagevt = uds_init_unix_listener(UDS_DIAG_ADDR, uds_event_diag_info)) == NULL)
|
||||
goto end1;
|
||||
|
||||
if ((logevt = uds_init_unix_listener(UDS_LOGLEVEL_UPD, uds_event_debug_level)) == NULL)
|
||||
goto end2;
|
||||
|
||||
if (chmod(UDS_BUILD_CONN_ADDR, UDS_FILE_MODE) < 0 ||
|
||||
chmod(UDS_DIAG_ADDR, UDS_FILE_MODE) < 0 ||
|
||||
chmod(UDS_LOGLEVEL_UPD, UDS_FILE_MODE) < 0) {
|
||||
uds_err("set sock file mode to %o failed, errno:%d", UDS_FILE_MODE, errno);
|
||||
goto end3;
|
||||
}
|
||||
|
||||
ret = uds_thread_execute();
|
||||
if (ret < 0) {
|
||||
uds_err("uds create thread failed.");
|
||||
}
|
||||
end3:
|
||||
uds_del_event(logevt);
|
||||
end2:
|
||||
uds_del_event(diagevt);
|
||||
end1:
|
||||
uds_del_event(tcpevt);
|
||||
end:
|
||||
uds_del_event(udsevt);
|
||||
rollbackefd:
|
||||
for (int i = 0; i < p_uds_var->work_thread_num; i++)
|
||||
close(p_uds_var->efd[i]);
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
int uds_env_prepare()
|
||||
{
|
||||
DIR *dir;
|
||||
if (access(UDS_BUILD_CONN_ADDR, 0) == 0)
|
||||
return EVENT_OK;
|
||||
|
||||
if ((dir = opendir(UDS_BUILD_CONN_DIR)) == NULL) {
|
||||
if (mkdir(UDS_BUILD_CONN_DIR, 0600) < 0) {
|
||||
uds_err("mkdir %s failed.", UDS_BUILD_CONN_DIR);
|
||||
return EVENT_ERR;
|
||||
}
|
||||
} else {
|
||||
closedir(dir);
|
||||
}
|
||||
return EVENT_OK;
|
||||
}
|
||||
|
||||
int uds_hash_init()
|
||||
{
|
||||
event_tmout_hash = g_hash_table_new(g_direct_hash, g_direct_equal);
|
||||
if (event_tmout_hash == NULL) {
|
||||
uds_err("g_hash_table_new failed.");
|
||||
return EVENT_ERR;
|
||||
}
|
||||
uds_log("init event time out hash");
|
||||
return EVENT_OK;
|
||||
}
|
||||
|
||||
void uds_hash_destroy()
|
||||
{
|
||||
g_hash_table_destroy(event_tmout_hash);
|
||||
event_tmout_hash = NULL;
|
||||
return;
|
||||
}
|
||||
|
||||
#pragma GCC diagnostic ignored "-Wint-to-pointer-cast"
|
||||
int uds_hash_insert_dirct(GHashTable *table, int key, struct uds_event *value)
|
||||
{
|
||||
if (g_hash_table_insert(table, (gpointer)key, value) == 0) {
|
||||
uds_err("Hash table key:%d is already exist, update it.", key);
|
||||
return -1;
|
||||
}
|
||||
uds_log("Hash insert key:%d", key);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void *uds_hash_lookup_dirct(GHashTable *table, int key)
|
||||
{
|
||||
return (void *)g_hash_table_lookup(table, (gpointer)key);
|
||||
}
|
||||
|
||||
int uds_hash_remove_dirct(GHashTable *table, int key)
|
||||
{
|
||||
if (g_hash_table_remove(table, (gpointer)key) == 0) {
|
||||
uds_err("Remove key:%d from hash failed.", key);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
#pragma GCC diagnostic pop
|
||||
|
||||
static void uds_rlimit()
|
||||
{
|
||||
struct rlimit lim;
|
||||
|
||||
getrlimit(RLIMIT_NOFILE, &lim);
|
||||
if (lim.rlim_cur >= UDS_FD_LIMIT)
|
||||
return;
|
||||
uds_log("uds proxy fd cur limit:%d, change to:%d", lim.rlim_cur, UDS_FD_LIMIT);
|
||||
lim.rlim_cur = UDS_FD_LIMIT;
|
||||
setrlimit(RLIMIT_NOFILE, &lim);
|
||||
return;
|
||||
}
|
||||
|
||||
// port invalid range 1024~65536
|
||||
#define UDS_PORT_VALID(port) (port >= 1024 && port < 65536)
|
||||
#define ARG_NUM 6
|
||||
static int uds_arg_check_and_init(int argc, char *argv[])
|
||||
{
|
||||
int myport;
|
||||
int peerport;
|
||||
memset(p_uds_var, 0, sizeof(struct uds_global_var));
|
||||
p_uds_var->loglevel = UDS_LOG_INFO;
|
||||
p_uds_var->logstr = uds_log_str;
|
||||
if (argc != ARG_NUM) {
|
||||
uds_helpinfo(argv);
|
||||
return -1;
|
||||
}
|
||||
myport = atoi(argv[3]);
|
||||
peerport = atoi(argv[5]);
|
||||
p_uds_var->work_thread_num = atoi(argv[1]);
|
||||
if (p_uds_var->work_thread_num <= 0 || p_uds_var->work_thread_num > UDS_WORK_THREAD_MAX) {
|
||||
uds_err("work thread num:%d is invalid.(must be 1~%d)", p_uds_var->work_thread_num, UDS_WORK_THREAD_MAX);
|
||||
return -1;
|
||||
}
|
||||
if (!UDS_PORT_VALID(myport) || !UDS_PORT_VALID(peerport)) {
|
||||
uds_err("local port:%d or peer port:%d invalid.(must be 1024~65535)", myport, peerport);
|
||||
return -1;
|
||||
}
|
||||
p_uds_var->efd = (int *)malloc(sizeof(int) * p_uds_var->work_thread_num);
|
||||
if (p_uds_var->efd == NULL) {
|
||||
uds_err("efd malloc failed, num:%d", p_uds_var->work_thread_num);
|
||||
return -1;
|
||||
}
|
||||
|
||||
p_uds_var->work_thread = (struct uds_thread_arg *)malloc(sizeof(struct uds_thread_arg) * p_uds_var->work_thread_num);
|
||||
if (p_uds_var->work_thread == NULL) {
|
||||
free(p_uds_var->efd);
|
||||
uds_err("work thread var malloc failed.");
|
||||
return -1;
|
||||
}
|
||||
#ifdef UDS_TEST_MODE
|
||||
p_uds_var->tcp.port = atoi(argv[3]);
|
||||
strncpy(p_uds_var->tcp.addr, argv[2], sizeof(p_uds_var->tcp.addr) - 1);
|
||||
p_uds_var->tcp.peerport = atoi(argv[5]);
|
||||
strncpy(p_uds_var->tcp.peeraddr, argv[4], sizeof(p_uds_var->tcp.peeraddr) - 1);
|
||||
|
||||
uds_log("uds proxy param thread num:%d ip:%s port:%u peerip:%s port:%u",
|
||||
p_uds_var->work_thread_num, p_uds_var->tcp.addr, p_uds_var->tcp.port,
|
||||
p_uds_var->tcp.peeraddr, p_uds_var->tcp.peerport);
|
||||
#else
|
||||
// vsock param: <thread num> <local cid> <local port> <peer cid> <peer port>
|
||||
// port and peerport is checked before
|
||||
p_uds_var->vsock.cid = atoi(argv[2]);
|
||||
p_uds_var->vsock.port = myport;
|
||||
p_uds_var->vsock.peercid = atoi(argv[4]);
|
||||
p_uds_var->vsock.peerport = peerport;
|
||||
#endif
|
||||
g_event_var = (struct uds_event_global_var *)malloc(sizeof(struct uds_event_global_var) * p_uds_var->work_thread_num);
|
||||
if (g_event_var == NULL) {
|
||||
free(p_uds_var->efd);
|
||||
free(p_uds_var->work_thread);
|
||||
uds_err("event variable malloc failed");
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void uds_sig_pipe(int signum)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
void uds_helpinfo(char *argv[])
|
||||
{
|
||||
uds_err("Usage:");
|
||||
uds_err(" %s <thread nums> <addr> <port> <peeraddr> <peerport>.", argv[0]);
|
||||
uds_err("Param:");
|
||||
uds_err(" <thread nums> - numbers of work thread(currently only supports 1 thread)");
|
||||
uds_err(" <addr> - server ip address");
|
||||
uds_err(" <port> - port number");
|
||||
uds_err(" <peeraddr> - peer address");
|
||||
uds_err(" <peerport> - peer port");
|
||||
return;
|
||||
}
|
||||
|
||||
int check_socket_lock(void)
|
||||
{
|
||||
int lock_fd = open(UDS_LOCK_ADDR, O_RDONLY | O_CREAT, 0600);
|
||||
if (lock_fd == -1)
|
||||
return -EINVAL;
|
||||
|
||||
return flock(lock_fd, LOCK_EX | LOCK_NB);
|
||||
}
|
||||
|
||||
#ifdef QTFS_SERVER
|
||||
int uds_proxy_main(int argc, char *argv[])
|
||||
{
|
||||
#else
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
mode_t newmask = 0077;
|
||||
uds_log("change uds umask from:%o to %o", umask(newmask), newmask);
|
||||
#endif
|
||||
if (uds_arg_check_and_init(argc, argv) < 0) {
|
||||
uds_err("global var init failed.");
|
||||
return -1;
|
||||
}
|
||||
if (uds_env_prepare() != EVENT_OK) {
|
||||
uds_err("proxy prepare environment failed.");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (check_socket_lock() < 0) {
|
||||
uds_err("another proxy running");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (uds_hash_init() != EVENT_OK) {
|
||||
uds_err("proxy hash init failed.");
|
||||
return -1;
|
||||
}
|
||||
#ifndef QTFS_SERVER
|
||||
uds_rlimit();
|
||||
#endif
|
||||
signal(SIGPIPE, uds_sig_pipe);
|
||||
uds_thread_create();
|
||||
uds_hash_destroy();
|
||||
|
||||
return 0;
|
||||
}
|
160
qtfs/ipc/uds_main.h
Normal file
160
qtfs/ipc/uds_main.h
Normal file
@ -0,0 +1,160 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#ifndef __QTFS_UDS_MAIN_H__
|
||||
#define __QTFS_UDS_MAIN_H__
|
||||
|
||||
#include <time.h>
|
||||
|
||||
#include "uds_module.h"
|
||||
|
||||
#define UDS_EPOLL_MAX_EVENTS 64
|
||||
#define UDS_WORK_THREAD_MAX 1 // Temporarily only supports 1 thread
|
||||
#define UDS_FD_LIMIT 65536
|
||||
#define UDS_FILE_MODE 0600 // for least privilege
|
||||
|
||||
extern struct uds_global_var *p_uds_var;
|
||||
extern GHashTable *event_tmout_hash;
|
||||
|
||||
enum {
|
||||
UDS_LOG_NONE,
|
||||
UDS_LOG_ERROR,
|
||||
UDS_LOG_INFO,
|
||||
UDS_LOG_MAX,
|
||||
};
|
||||
|
||||
#define uds_log(info, ...) \
|
||||
if (p_uds_var->loglevel >= UDS_LOG_INFO) {\
|
||||
time_t t; \
|
||||
struct tm p; \
|
||||
time(&t); \
|
||||
localtime_r(&t, &p); \
|
||||
printf("[%d/%02d/%02d %02d:%02d:%02d][LOG:%s:%3d]"info"\n", \
|
||||
p.tm_year + 1900, p.tm_mon+1, p.tm_mday, \
|
||||
p.tm_hour, p.tm_min, p.tm_sec, __func__, __LINE__, ##__VA_ARGS__); \
|
||||
}
|
||||
|
||||
#define uds_err(info, ...) \
|
||||
if (p_uds_var->loglevel >= UDS_LOG_ERROR) {\
|
||||
time_t t; \
|
||||
struct tm p; \
|
||||
time(&t); \
|
||||
localtime_r(&t, &p); \
|
||||
printf("[%d/%02d/%02d %02d:%02d:%02d][ERROR:%s:%3d]"info"\n", \
|
||||
p.tm_year + 1900, p.tm_mon+1, p.tm_mday, \
|
||||
p.tm_hour, p.tm_min, p.tm_sec, __func__, __LINE__, ##__VA_ARGS__); \
|
||||
}
|
||||
|
||||
enum {
|
||||
UDS_THREAD_EPWAIT = 1, // epoll wait status
|
||||
};
|
||||
struct uds_thread_info {
|
||||
int fdnum;
|
||||
|
||||
int events;
|
||||
int status;
|
||||
};
|
||||
|
||||
struct uds_event_global_var {
|
||||
int cur;
|
||||
struct uds_event *tofree[UDS_EPOLL_MAX_EVENTS];
|
||||
char *msg_control;
|
||||
int msg_controllen;
|
||||
char *msg_control_send;
|
||||
int msg_controlsendlen;
|
||||
char *iov_base;
|
||||
int iov_len;
|
||||
char *iov_base_send;
|
||||
int iov_sendlen;
|
||||
char *buf;
|
||||
int buflen;
|
||||
};
|
||||
|
||||
struct uds_event {
|
||||
int fd; /* 本事件由这个fd触发 */
|
||||
unsigned int tofree : 1, /* 1--in to free list; 0--not */
|
||||
pipe : 1, // this is a pipe event
|
||||
tmout : 4,
|
||||
reserved : 26;
|
||||
union {
|
||||
struct uds_event *peer; /* peer event */
|
||||
int peerfd; // scm pipe 场景单向导通,只需要一个fd即可
|
||||
};
|
||||
int (*handler)(void *, int, struct uds_event_global_var *); /* event处理函数 */
|
||||
void *priv; // private data
|
||||
char cpath[UDS_SUN_PATH_LEN];
|
||||
char spath[UDS_SUN_PATH_LEN];
|
||||
};
|
||||
|
||||
|
||||
struct uds_thread_arg {
|
||||
int efd;
|
||||
struct uds_event_global_var *p_event_var;
|
||||
struct uds_thread_info info;
|
||||
};
|
||||
|
||||
struct uds_global_var {
|
||||
int work_thread_num;
|
||||
int *efd;
|
||||
struct uds_thread_arg *work_thread;
|
||||
int loglevel;
|
||||
char **logstr;
|
||||
#ifdef UDS_TEST_MODE
|
||||
struct _tcp {
|
||||
char addr[20];
|
||||
unsigned short port;
|
||||
char peeraddr[20];
|
||||
unsigned short peerport;
|
||||
} tcp;
|
||||
#else
|
||||
struct _vsock {
|
||||
unsigned int cid;
|
||||
unsigned int port;
|
||||
unsigned int peercid;
|
||||
unsigned int peerport;
|
||||
} vsock;
|
||||
#endif
|
||||
};
|
||||
enum uds_cs {
|
||||
UDS_SOCKET_CLIENT = 1,
|
||||
UDS_SOCKET_SERVER,
|
||||
};
|
||||
|
||||
struct uds_conn_arg {
|
||||
int cs; // client(1) or server(2)
|
||||
|
||||
int udstype; // DGRAM or STREAM
|
||||
char sun_path[UDS_SUN_PATH_LEN];
|
||||
int sockfd;
|
||||
int connfd;
|
||||
};
|
||||
|
||||
struct uds_event *uds_add_event(int fd, struct uds_event *peer, int (*handler)(void *, int, struct uds_event_global_var *), void *priv);
|
||||
struct uds_event *uds_add_pipe_event(int fd, int peerfd, int (*handler)(void *, int, struct uds_event_global_var *), void *priv);
|
||||
int uds_sock_step_accept(int sockFd, int family);
|
||||
int uds_build_tcp_connection(struct uds_conn_arg *arg);
|
||||
int uds_build_unix_connection(struct uds_conn_arg *arg);
|
||||
void uds_del_event(struct uds_event *evt);
|
||||
int uds_event_suspend(int efd, struct uds_event *event);
|
||||
int uds_event_insert(int efd, struct uds_event *event);
|
||||
int uds_hash_insert_dirct(GHashTable *table, int key, struct uds_event *value);
|
||||
void *uds_hash_lookup_dirct(GHashTable *table, int key);
|
||||
int uds_hash_remove_dirct(GHashTable *table, int key);
|
||||
int uds_recv_with_timeout(int fd, char *msg, int len);
|
||||
|
||||
#ifdef QTFS_SERVER
|
||||
int uds_proxy_main(int argc, char *argv[]);
|
||||
#endif
|
||||
#endif
|
37
qtfs/ipc/uds_module.h
Normal file
37
qtfs/ipc/uds_module.h
Normal file
@ -0,0 +1,37 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#ifndef __QTFS_UDS_MODULE_H__
|
||||
#define __QTFS_UDS_MODULE_H__
|
||||
|
||||
#define UDS_BUILD_CONN_ADDR "/var/run/qtfs/remote_uds.sock"
|
||||
#define UDS_DIAG_ADDR "/var/run/qtfs/uds_proxy_diag.sock"
|
||||
#define UDS_LOGLEVEL_UPD "/var/run/qtfs/uds_loglevel.sock"
|
||||
#define UDS_LOCK_ADDR "/var/run/qtfs/uds.lock"
|
||||
#define UDS_BUILD_CONN_DIR "/var/run/qtfs/"
|
||||
|
||||
#define UDS_PROXY_SUFFIX ".proxy"
|
||||
|
||||
#define UDS_SUN_PATH_LEN 108 // from glibc
|
||||
struct uds_proxy_remote_conn_req {
|
||||
unsigned short type;
|
||||
unsigned short resv;
|
||||
char sun_path[UDS_SUN_PATH_LEN];
|
||||
};
|
||||
struct uds_proxy_remote_conn_rsp {
|
||||
int ret;
|
||||
};
|
||||
|
||||
#endif
|
75
qtfs/qtfs/CMakeLists.txt
Normal file
75
qtfs/qtfs/CMakeLists.txt
Normal file
@ -0,0 +1,75 @@
|
||||
# Build Kernel modules
|
||||
set(MODULE_NAME qtfs)
|
||||
|
||||
## First, find the kernel build directory
|
||||
execute_process(
|
||||
COMMAND uname -r
|
||||
OUTPUT_VARIABLE KERNEL_RELEASE
|
||||
OUTPUT_STRIP_TRAILING_WHITESPACE)
|
||||
set(KBUILD_DIR /lib/modules/${KERNEL_RELEASE}/build/)
|
||||
find_file(KERNEL_MAKEFILE NAMES Makefile PATHS ${KBUILD_DIR} NO_DEFAULT_PATH)
|
||||
if (NOT KERNEL_MAKEFILE)
|
||||
message(FATAL_ERROR "There is no Makefile in ${KBUILD_DIR}!")
|
||||
endif ()
|
||||
|
||||
## Second, gather the source files
|
||||
set(COMM_SRC_FILES
|
||||
${QTFS_BASE_DIR}/qtfs_common/conn.c
|
||||
${QTFS_BASE_DIR}/qtfs_common/misc.c
|
||||
${QTFS_BASE_DIR}/qtfs_common/qtfs_check.c
|
||||
${QTFS_BASE_DIR}/qtfs_common/socket.c
|
||||
${QTFS_BASE_DIR}/qtfs_common/symbol_wrapper.c)
|
||||
|
||||
set(QTFS_SRC_FILES
|
||||
${QTFS_BASE_DIR}/qtfs/fifo.c
|
||||
${QTFS_BASE_DIR}/qtfs/miss.c
|
||||
${QTFS_BASE_DIR}/qtfs/proc.c
|
||||
${QTFS_BASE_DIR}/qtfs/qtfs-mod.c
|
||||
${QTFS_BASE_DIR}/qtfs/sb.c
|
||||
${QTFS_BASE_DIR}/qtfs/syscall.c
|
||||
${QTFS_BASE_DIR}/qtfs/xattr.c)
|
||||
|
||||
set(QTFS_MODULE_SRC ${COMM_SRC_FILES} ${QTFS_SRC_FILES} ${QTFS_BASE_DIR}/utils/utils.c)
|
||||
|
||||
## Third, make the src files accessible to the kernel Makefile
|
||||
set(MODULE_BUILD_DIR ${CMAKE_CURRENT_BINARY_DIR}/)
|
||||
set(QTFS_KBUILD_FILE "obj-m := ${MODULE_NAME}.o")
|
||||
foreach (file ${QTFS_MODULE_SRC})
|
||||
file(RELATIVE_PATH file "${CMAKE_CURRENT_SOURCE_DIR}" "${file}")
|
||||
configure_file(${file} ${file} COPYONLY)
|
||||
string(REPLACE ".c" ".o" file_obj "${file}")
|
||||
set(QTFS_KBUILD_FILE "${QTFS_KBUILD_FILE}\n${MODULE_NAME}-y += ${file_obj}")
|
||||
endforeach ()
|
||||
|
||||
set(QTFS_KBUILD_FILE "${QTFS_KBUILD_FILE}\nccflags-y += -I${QTFS_BASE_DIR}/include")
|
||||
set(QTFS_KBUILD_FILE "${QTFS_KBUILD_FILE}\nccflags-y += -I${QTFS_BASE_DIR}/ipc")
|
||||
set(QTFS_KBUILD_FILE "${QTFS_KBUILD_FILE}\nccflags-y += -I${QTFS_BASE_DIR}/qtfs")
|
||||
set(QTFS_KBUILD_FILE "${QTFS_KBUILD_FILE}\nccflags-y += -I${QTFS_BASE_DIR}/utils")
|
||||
set(QTFS_KBUILD_FILE "${QTFS_KBUILD_FILE}\nccflags-y += -DQTFS_CLIENT")
|
||||
|
||||
if (DEFINED UDS_TEST_MODE OR DEFINED QTFS_TEST_MODE)
|
||||
set(QTFS_KBUILD_FILE "${QTFS_KBUILD_FILE}\nccflags-y += -DQTFS_TEST_MODE")
|
||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUDS_TEST_MODE")
|
||||
endif ()
|
||||
|
||||
## Forth, generate a Kbuild file
|
||||
file(WRITE ${CMAKE_CURRENT_BINARY_DIR}/Kbuild ${QTFS_KBUILD_FILE})
|
||||
|
||||
## Fifth, add a custom target to build the module
|
||||
set(MODULE_CMD ${CMAKE_MAKE_PROGRAM} -C ${KBUILD_DIR} M=${CMAKE_CURRENT_BINARY_DIR})
|
||||
add_custom_command(OUTPUT ${MODULE_NAME}.ko
|
||||
COMMAND ${MODULE_CMD} modules
|
||||
WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
|
||||
DEPENDS ${QTFS_MODULE_SRC} ${CMAKE_CURRENT_BINARY_DIR}/Kbuild
|
||||
VERBATIM)
|
||||
add_custom_target(${MODULE_NAME} DEPENDS ${MODULE_NAME}.ko)
|
||||
add_custom_target(${MODULE_NAME}-clean COMMAND ${MODULE_CMD} clean)
|
||||
|
||||
# This target helps parsing C files for IDEs like CLion
|
||||
add_library(dummy-${MODULE_NAME} EXCLUDE_FROM_ALL ${QTFS_MODULE_SRC})
|
||||
target_include_directories(dummy-${MODULE_NAME} PRIVATE
|
||||
${KBUILD_DIR}/include
|
||||
${QTFS_BASE_DIR}/include
|
||||
${QTFS_BASE_DIR}/ipc
|
||||
${QTFS_BASE_DIR}/qtfs)
|
||||
set_target_properties(dummy-${MODULE_NAME} PROPERTIES DEPRECATION "DO NOT BUILD THIS TARGET")
|
340
qtfs/qtfs/License
Normal file
340
qtfs/qtfs/License
Normal file
@ -0,0 +1,340 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 2, June 1991
|
||||
|
||||
Copyright (C) 1989, 1991 Free Software Foundation, Inc.
|
||||
51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The licenses for most software are designed to take away your
|
||||
freedom to share and change it. By contrast, the GNU General Public
|
||||
License is intended to guarantee your freedom to share and change free
|
||||
software--to make sure the software is free for all its users. This
|
||||
General Public License applies to most of the Free Software
|
||||
Foundation's software and to any other program whose authors commit to
|
||||
using it. (Some other Free Software Foundation software is covered by
|
||||
the GNU Library General Public License instead.) You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
this service if you wish), that you receive source code or can get it
|
||||
if you want it, that you can change the software or use pieces of it
|
||||
in new free programs; and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to make restrictions that forbid
|
||||
anyone to deny you these rights or to ask you to surrender the rights.
|
||||
These restrictions translate to certain responsibilities for you if you
|
||||
distribute copies of the software, or if you modify it.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must give the recipients all the rights that
|
||||
you have. You must make sure that they, too, receive or can get the
|
||||
source code. And you must show them these terms so they know their
|
||||
rights.
|
||||
|
||||
We protect your rights with two steps: (1) copyright the software, and
|
||||
(2) offer you this license which gives you legal permission to copy,
|
||||
distribute and/or modify the software.
|
||||
|
||||
Also, for each author's protection and ours, we want to make certain
|
||||
that everyone understands that there is no warranty for this free
|
||||
software. If the software is modified by someone else and passed on, we
|
||||
want its recipients to know that what they have is not the original, so
|
||||
that any problems introduced by others will not reflect on the original
|
||||
authors' reputations.
|
||||
|
||||
Finally, any free program is threatened constantly by software
|
||||
patents. We wish to avoid the danger that redistributors of a free
|
||||
program will individually obtain patent licenses, in effect making the
|
||||
program proprietary. To prevent this, we have made it clear that any
|
||||
patent must be licensed for everyone's free use or not licensed at all.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||
|
||||
0. This License applies to any program or other work which contains
|
||||
a notice placed by the copyright holder saying it may be distributed
|
||||
under the terms of this General Public License. The "Program", below,
|
||||
refers to any such program or work, and a "work based on the Program"
|
||||
means either the Program or any derivative work under copyright law:
|
||||
that is to say, a work containing the Program or a portion of it,
|
||||
either verbatim or with modifications and/or translated into another
|
||||
language. (Hereinafter, translation is included without limitation in
|
||||
the term "modification".) Each licensee is addressed as "you".
|
||||
|
||||
Activities other than copying, distribution and modification are not
|
||||
covered by this License; they are outside its scope. The act of
|
||||
running the Program is not restricted, and the output from the Program
|
||||
is covered only if its contents constitute a work based on the
|
||||
Program (independent of having been made by running the Program).
|
||||
Whether that is true depends on what the Program does.
|
||||
|
||||
1. You may copy and distribute verbatim copies of the Program's
|
||||
source code as you receive it, in any medium, provided that you
|
||||
conspicuously and appropriately publish on each copy an appropriate
|
||||
copyright notice and disclaimer of warranty; keep intact all the
|
||||
notices that refer to this License and to the absence of any warranty;
|
||||
and give any other recipients of the Program a copy of this License
|
||||
along with the Program.
|
||||
|
||||
You may charge a fee for the physical act of transferring a copy, and
|
||||
you may at your option offer warranty protection in exchange for a fee.
|
||||
|
||||
2. You may modify your copy or copies of the Program or any portion
|
||||
of it, thus forming a work based on the Program, and copy and
|
||||
distribute such modifications or work under the terms of Section 1
|
||||
above, provided that you also meet all of these conditions:
|
||||
|
||||
a) You must cause the modified files to carry prominent notices
|
||||
stating that you changed the files and the date of any change.
|
||||
|
||||
b) You must cause any work that you distribute or publish, that in
|
||||
whole or in part contains or is derived from the Program or any
|
||||
part thereof, to be licensed as a whole at no charge to all third
|
||||
parties under the terms of this License.
|
||||
|
||||
c) If the modified program normally reads commands interactively
|
||||
when run, you must cause it, when started running for such
|
||||
interactive use in the most ordinary way, to print or display an
|
||||
announcement including an appropriate copyright notice and a
|
||||
notice that there is no warranty (or else, saying that you provide
|
||||
a warranty) and that users may redistribute the program under
|
||||
these conditions, and telling the user how to view a copy of this
|
||||
License. (Exception: if the Program itself is interactive but
|
||||
does not normally print such an announcement, your work based on
|
||||
the Program is not required to print an announcement.)
|
||||
|
||||
These requirements apply to the modified work as a whole. If
|
||||
identifiable sections of that work are not derived from the Program,
|
||||
and can be reasonably considered independent and separate works in
|
||||
themselves, then this License, and its terms, do not apply to those
|
||||
sections when you distribute them as separate works. But when you
|
||||
distribute the same sections as part of a whole which is a work based
|
||||
on the Program, the distribution of the whole must be on the terms of
|
||||
this License, whose permissions for other licensees extend to the
|
||||
entire whole, and thus to each and every part regardless of who wrote it.
|
||||
|
||||
Thus, it is not the intent of this section to claim rights or contest
|
||||
your rights to work written entirely by you; rather, the intent is to
|
||||
exercise the right to control the distribution of derivative or
|
||||
collective works based on the Program.
|
||||
|
||||
In addition, mere aggregation of another work not based on the Program
|
||||
with the Program (or with a work based on the Program) on a volume of
|
||||
a storage or distribution medium does not bring the other work under
|
||||
the scope of this License.
|
||||
|
||||
3. You may copy and distribute the Program (or a work based on it,
|
||||
under Section 2) in object code or executable form under the terms of
|
||||
Sections 1 and 2 above provided that you also do one of the following:
|
||||
|
||||
a) Accompany it with the complete corresponding machine-readable
|
||||
source code, which must be distributed under the terms of Sections
|
||||
1 and 2 above on a medium customarily used for software interchange; or,
|
||||
|
||||
b) Accompany it with a written offer, valid for at least three
|
||||
years, to give any third party, for a charge no more than your
|
||||
cost of physically performing source distribution, a complete
|
||||
machine-readable copy of the corresponding source code, to be
|
||||
distributed under the terms of Sections 1 and 2 above on a medium
|
||||
customarily used for software interchange; or,
|
||||
|
||||
c) Accompany it with the information you received as to the offer
|
||||
to distribute corresponding source code. (This alternative is
|
||||
allowed only for noncommercial distribution and only if you
|
||||
received the program in object code or executable form with such
|
||||
an offer, in accord with Subsection b above.)
|
||||
|
||||
The source code for a work means the preferred form of the work for
|
||||
making modifications to it. For an executable work, complete source
|
||||
code means all the source code for all modules it contains, plus any
|
||||
associated interface definition files, plus the scripts used to
|
||||
control compilation and installation of the executable. However, as a
|
||||
special exception, the source code distributed need not include
|
||||
anything that is normally distributed (in either source or binary
|
||||
form) with the major components (compiler, kernel, and so on) of the
|
||||
operating system on which the executable runs, unless that component
|
||||
itself accompanies the executable.
|
||||
|
||||
If distribution of executable or object code is made by offering
|
||||
access to copy from a designated place, then offering equivalent
|
||||
access to copy the source code from the same place counts as
|
||||
distribution of the source code, even though third parties are not
|
||||
compelled to copy the source along with the object code.
|
||||
|
||||
4. You may not copy, modify, sublicense, or distribute the Program
|
||||
except as expressly provided under this License. Any attempt
|
||||
otherwise to copy, modify, sublicense or distribute the Program is
|
||||
void, and will automatically terminate your rights under this License.
|
||||
However, parties who have received copies, or rights, from you under
|
||||
this License will not have their licenses terminated so long as such
|
||||
parties remain in full compliance.
|
||||
|
||||
5. You are not required to accept this License, since you have not
|
||||
signed it. However, nothing else grants you permission to modify or
|
||||
distribute the Program or its derivative works. These actions are
|
||||
prohibited by law if you do not accept this License. Therefore, by
|
||||
modifying or distributing the Program (or any work based on the
|
||||
Program), you indicate your acceptance of this License to do so, and
|
||||
all its terms and conditions for copying, distributing or modifying
|
||||
the Program or works based on it.
|
||||
|
||||
6. Each time you redistribute the Program (or any work based on the
|
||||
Program), the recipient automatically receives a license from the
|
||||
original licensor to copy, distribute or modify the Program subject to
|
||||
these terms and conditions. You may not impose any further
|
||||
restrictions on the recipients' exercise of the rights granted herein.
|
||||
You are not responsible for enforcing compliance by third parties to
|
||||
this License.
|
||||
|
||||
7. If, as a consequence of a court judgment or allegation of patent
|
||||
infringement or for any other reason (not limited to patent issues),
|
||||
conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot
|
||||
distribute so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you
|
||||
may not distribute the Program at all. For example, if a patent
|
||||
license would not permit royalty-free redistribution of the Program by
|
||||
all those who receive copies directly or indirectly through you, then
|
||||
the only way you could satisfy both it and this License would be to
|
||||
refrain entirely from distribution of the Program.
|
||||
|
||||
If any portion of this section is held invalid or unenforceable under
|
||||
any particular circumstance, the balance of the section is intended to
|
||||
apply and the section as a whole is intended to apply in other
|
||||
circumstances.
|
||||
|
||||
It is not the purpose of this section to induce you to infringe any
|
||||
patents or other property right claims or to contest validity of any
|
||||
such claims; this section has the sole purpose of protecting the
|
||||
integrity of the free software distribution system, which is
|
||||
implemented by public license practices. Many people have made
|
||||
generous contributions to the wide range of software distributed
|
||||
through that system in reliance on consistent application of that
|
||||
system; it is up to the author/donor to decide if he or she is willing
|
||||
to distribute software through any other system and a licensee cannot
|
||||
impose that choice.
|
||||
|
||||
This section is intended to make thoroughly clear what is believed to
|
||||
be a consequence of the rest of this License.
|
||||
|
||||
8. If the distribution and/or use of the Program is restricted in
|
||||
certain countries either by patents or by copyrighted interfaces, the
|
||||
original copyright holder who places the Program under this License
|
||||
may add an explicit geographical distribution limitation excluding
|
||||
those countries, so that distribution is permitted only in or among
|
||||
countries not thus excluded. In such case, this License incorporates
|
||||
the limitation as if written in the body of this License.
|
||||
|
||||
9. The Free Software Foundation may publish revised and/or new versions
|
||||
of the General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the Program
|
||||
specifies a version number of this License which applies to it and "any
|
||||
later version", you have the option of following the terms and conditions
|
||||
either of that version or of any later version published by the Free
|
||||
Software Foundation. If the Program does not specify a version number of
|
||||
this License, you may choose any version ever published by the Free Software
|
||||
Foundation.
|
||||
|
||||
10. If you wish to incorporate parts of the Program into other free
|
||||
programs whose distribution conditions are different, write to the author
|
||||
to ask for permission. For software which is copyrighted by the Free
|
||||
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||
make exceptions for this. Our decision will be guided by the two goals
|
||||
of preserving the free status of all derivatives of our free software and
|
||||
of promoting the sharing and reuse of software generally.
|
||||
|
||||
NO WARRANTY
|
||||
|
||||
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||
REPAIR OR CORRECTION.
|
||||
|
||||
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||
POSSIBILITY OF SUCH DAMAGES.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
convey the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation; either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program; if not, write to the Free Software
|
||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program is interactive, make it output a short notice like this
|
||||
when it starts in an interactive mode:
|
||||
|
||||
Gnomovision version 69, Copyright (C) year name of author
|
||||
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, the commands you use may
|
||||
be called something other than `show w' and `show c'; they could even be
|
||||
mouse-clicks or menu items--whatever suits your program.
|
||||
|
||||
You should also get your employer (if you work as a programmer) or your
|
||||
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||
necessary. Here is a sample; alter the names:
|
||||
|
||||
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||
|
||||
<signature of Ty Coon>, 1 April 1989
|
||||
Ty Coon, President of Vice
|
||||
|
||||
This General Public License does not permit incorporating your program into
|
||||
proprietary programs. If your program is a subroutine library, you may
|
||||
consider it more useful to permit linking proprietary applications with the
|
||||
library. If this is what you want to do, use the GNU Library General
|
||||
Public License instead of this License.
|
27
qtfs/qtfs/Makefile
Normal file
27
qtfs/qtfs/Makefile
Normal file
@ -0,0 +1,27 @@
|
||||
ifdef QTFS_TEST_MODE
|
||||
ccflags-y += -I$(src)/../ -I$(src)/../utils/ -I$(src)/../include/ -I$(src)/../ipc/ -I$(src) -DQTFS_CLIENT -DQTFS_TEST_MODE
|
||||
else
|
||||
ccflags-y += -I$(src)/../ -I$(src)/../utils/ -I$(src)/../include/ -I$(src)/../ipc/ -I$(src) -DQTFS_CLIENT
|
||||
endif
|
||||
|
||||
KBUILD=/lib/modules/$(shell uname -r)/build/
|
||||
COMM=../qtfs_common/
|
||||
COMMO=$(COMM)/conn.o $(COMM)/misc.o $(COMM)/symbol_wrapper.o $(COMM)/socket.o
|
||||
|
||||
obj-m:=qtfs.o
|
||||
qtfs-objs:=qtfs-mod.o sb.o syscall.o xattr.o proc.o miss.o fifo.o $(COMMO) ../utils/utils.o
|
||||
|
||||
all: qtfs
|
||||
|
||||
qtfs:
|
||||
make -C $(KBUILD) M=$(PWD) modules
|
||||
@test -z $(QTFS_TEST_MODE) || echo "Important risk warning: The test mode is turned on,\
|
||||
and qtfs will expose the network port, which will bring security risks and is only for\
|
||||
testing! If you do not understand the risks, please don't use or compile again without\
|
||||
QTFS_TEST_MODE."
|
||||
|
||||
clean:
|
||||
make -C $(KBUILD) M=$(PWD) clean
|
||||
rm -rf ../*.o ../.*.o.cmd
|
||||
rm -rf $(COMMO) $(COMM).*.o.cmd
|
||||
rm -rf ../utils/*.o ../utils/.*.o.cmd
|
221
qtfs/qtfs/fifo.c
Normal file
221
qtfs/qtfs/fifo.c
Normal file
@ -0,0 +1,221 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/time.h>
|
||||
#include <linux/fs_struct.h>
|
||||
#include <linux/statfs.h>
|
||||
#include <linux/wait.h>
|
||||
#include <linux/version.h>
|
||||
#include <asm-generic/ioctls.h>
|
||||
#include <asm-generic/termbits.h>
|
||||
|
||||
#include "conn.h"
|
||||
#include "qtfs-mod.h"
|
||||
#include "req.h"
|
||||
#include "log.h"
|
||||
|
||||
// 对接rust,长度对齐1字节
|
||||
#pragma pack(1)
|
||||
struct qtreq_fifo_open {
|
||||
u64 flags;
|
||||
u32 mode;
|
||||
char path[MAX_PATH_LEN];
|
||||
};
|
||||
|
||||
struct qtrsp_fifo_open {
|
||||
s32 errno;
|
||||
};
|
||||
|
||||
struct qtreq_fifo_read {
|
||||
u64 len;
|
||||
};
|
||||
|
||||
struct qtrsp_fifo_read {
|
||||
s32 errno; // same as kernel errcode, 0 is ok, < 0 is errcode
|
||||
u64 len;
|
||||
};
|
||||
|
||||
struct qtreq_fifo_write {
|
||||
u64 len;
|
||||
};
|
||||
|
||||
struct qtrsp_fifo_write {
|
||||
s32 errno;
|
||||
u64 len;
|
||||
};
|
||||
|
||||
struct qtreq_fifo_close {
|
||||
// nothing
|
||||
};
|
||||
|
||||
struct qtrsp_fifo_close {
|
||||
// nothing
|
||||
};
|
||||
#pragma pack()
|
||||
|
||||
static void qtfs_fifo_put_file(struct file *file)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = file->private_data;
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("fifo private data invalid to put");
|
||||
return;
|
||||
}
|
||||
qtfs_fifo_put_param(pvar);
|
||||
file->private_data = NULL;
|
||||
return;
|
||||
}
|
||||
|
||||
int qtfs_fifo_open(struct inode *inode, struct file *file)
|
||||
{
|
||||
struct kvec vec_save;
|
||||
unsigned int sendmax_save;
|
||||
struct qtreq *req;
|
||||
struct qtreq_fifo_open *fiforeq;
|
||||
struct qtrsp_fifo_open *rsp;
|
||||
struct qtfs_conn_var_s *pvar = NULL;
|
||||
int ret;
|
||||
|
||||
req = (struct qtreq *)kmalloc(sizeof(struct qtreq) + sizeof(struct qtreq_fifo_open), GFP_KERNEL);
|
||||
if (req == NULL) {
|
||||
qtfs_err("get fifo open memory failed.");
|
||||
return -ENOMEM;
|
||||
}
|
||||
memset(req, 0, sizeof(struct qtreq) + sizeof(struct qtreq_fifo_open));
|
||||
|
||||
pvar = qtfs_fifo_get_param();
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("fifo get param failed.");
|
||||
kfree(req);
|
||||
return -EINVAL;
|
||||
}
|
||||
fiforeq = (struct qtreq_fifo_open *)req->data;
|
||||
|
||||
if (qtfs_fullname(fiforeq->path, file->f_path.dentry, sizeof(fiforeq->path)) < 0) {
|
||||
qtfs_err("qtfs fifo fullname failed");
|
||||
kfree(req);
|
||||
qtfs_fifo_put_param(pvar);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
fiforeq->flags = file->f_flags;
|
||||
fiforeq->mode = file->f_mode;
|
||||
qtfs_err("fifo open path:%s size req:%u size open:%u, flags:%lu mode%u",
|
||||
fiforeq->path, sizeof(struct qtreq), QTFS_SEND_SIZE(struct qtreq_fifo_open, fiforeq->path),
|
||||
fiforeq->flags, fiforeq->mode);
|
||||
vec_save = pvar->vec_send;
|
||||
sendmax_save = pvar->send_max;
|
||||
pvar->vec_send.iov_base = req;
|
||||
pvar->send_max = sizeof(struct qtreq) + sizeof(struct qtreq_fifo_open);
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_OPEN, QTFS_SEND_SIZE(struct qtreq_fifo_open, fiforeq->path));
|
||||
pvar->vec_send = vec_save;
|
||||
pvar->send_max = sendmax_save;
|
||||
|
||||
if (IS_ERR_OR_NULL(rsp) || rsp->errno != 0) {
|
||||
ret = IS_ERR_OR_NULL(rsp) ? -EFAULT : -rsp->errno;
|
||||
qtfs_fifo_put_param(pvar);
|
||||
qtfs_err("qtfs fifo open :%s failed mode:%o flag:%x", fiforeq->path, fiforeq->mode, fiforeq->flags);
|
||||
kfree(req);
|
||||
return ret;
|
||||
}
|
||||
kfree(req);
|
||||
WARN_ON(file->private_data);
|
||||
file->private_data = pvar;
|
||||
return 0;
|
||||
}
|
||||
|
||||
ssize_t qtfs_fifo_readiter(struct kiocb *kio, struct iov_iter *iov)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = kio->ki_filp->private_data;
|
||||
struct qtreq_fifo_read *req;
|
||||
struct qtrsp_fifo_read *rsp;
|
||||
int total = 0;
|
||||
int ret;
|
||||
|
||||
if (pvar == NULL || !virt_addr_valid(pvar)) {
|
||||
qtfs_err("invalid fifo read req, private data is invalid");
|
||||
return -EFAULT;
|
||||
}
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
req->len = iov_iter_count(iov);
|
||||
pvar->vec_recv.iov_len = QTFS_MSG_HEAD_LEN + sizeof(struct qtrsp_fifo_read);
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_READITER, sizeof(struct qtreq_fifo_read));
|
||||
if (IS_ERR_OR_NULL(rsp) || rsp->errno != 0) {
|
||||
qtfs_err("remote run failed. or errno:%d", (rsp == NULL) ? -1 : rsp->errno);
|
||||
return -EFAULT;
|
||||
}
|
||||
|
||||
while (total < rsp->len) {
|
||||
ret = pvar->conn_ops->conn_recv_iter(&pvar->conn_var, iov, false);
|
||||
if (ret <= 0) {
|
||||
qtfs_err("recv iter from conn module ret:%d", ret);
|
||||
break;
|
||||
}
|
||||
total += ret;
|
||||
}
|
||||
return total;
|
||||
}
|
||||
|
||||
ssize_t qtfs_fifo_writeiter(struct kiocb *kio, struct iov_iter *iov)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = kio->ki_filp->private_data;
|
||||
struct qtreq_fifo_write *req;
|
||||
struct qtrsp_fifo_write *rsp;
|
||||
int total = 0;
|
||||
|
||||
if (pvar == NULL || !virt_addr_valid(pvar)) {
|
||||
qtfs_err("invalid fifo write req, private data is invalid");
|
||||
return -EFAULT;
|
||||
}
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
req->len = iov_iter_count(iov);
|
||||
pvar->vec_recv.iov_len = QTFS_MSG_HEAD_LEN + sizeof(struct qtrsp_fifo_write);
|
||||
pvar->iov_send = iov;
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_WRITE, sizeof(struct qtreq_fifo_write));
|
||||
if (IS_ERR_OR_NULL(rsp) || rsp->errno != 0) {
|
||||
qtfs_err("fifo write remote run failed, or errno:%d", (rsp == NULL) ? -1 : rsp->errno);
|
||||
return -EFAULT;
|
||||
}
|
||||
return rsp->len;
|
||||
}
|
||||
|
||||
int qtfs_fifo_release(struct inode *inode, struct file *file)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = file->private_data;
|
||||
struct qtrsp_fifo_close *rsp = NULL;
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("invalid fifo write req, private data is invalid");
|
||||
return -EFAULT;
|
||||
}
|
||||
pvar->vec_recv.iov_len = QTFS_MSG_HEAD_LEN;
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_CLOSE, 0);
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
qtfs_err("fifo close failed");
|
||||
}
|
||||
qtfs_fifo_put_file(file);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static __poll_t
|
||||
qtfs_fifo_poll(struct file *filp, poll_table *wait)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct file_operations qtfsfifo_ops = {
|
||||
.read_iter = qtfs_fifo_readiter,
|
||||
.write_iter = qtfs_fifo_writeiter,
|
||||
.open = qtfs_fifo_open,
|
||||
.release = qtfs_fifo_release,
|
||||
.llseek = no_llseek,
|
||||
.poll = qtfs_fifo_poll,
|
||||
};
|
82
qtfs/qtfs/miss.c
Normal file
82
qtfs/qtfs/miss.c
Normal file
@ -0,0 +1,82 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/time.h>
|
||||
#include <linux/fs_struct.h>
|
||||
#include <linux/statfs.h>
|
||||
#include <linux/pagemap.h>
|
||||
#include <linux/mpage.h>
|
||||
#include <linux/wait.h>
|
||||
#include "conn.h"
|
||||
#include "qtfs-mod.h"
|
||||
#include "req.h"
|
||||
|
||||
#include "log.h"
|
||||
#include "ops.h"
|
||||
|
||||
static int miss_open(struct qtreq *miss)
|
||||
{
|
||||
struct qtrsp_open *missrsp = (struct qtrsp_open *)miss->data;
|
||||
struct qtreq_close *req;
|
||||
struct qtrsp_close *rsp;
|
||||
struct qtfs_conn_var_s *pvar = NULL;
|
||||
if (missrsp == NULL) {
|
||||
qtfs_err("input response is NULL.");
|
||||
return QTFS_ERR;
|
||||
}
|
||||
if (missrsp->ret == QTFS_ERR)
|
||||
return QTFS_OK; // no need to close
|
||||
|
||||
pvar = qtfs_conn_get_param();
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("qtfs miss open pvar invalid.");
|
||||
return QTFS_ERR;
|
||||
}
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
req->fd = missrsp->fd;
|
||||
qtfs_err("miss open proc fd:%d.", req->fd);
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_CLOSE, sizeof(struct qtreq_close));
|
||||
qtfs_conn_put_param(pvar);
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
return QTFS_ERR;
|
||||
}
|
||||
|
||||
return QTFS_OK;
|
||||
}
|
||||
|
||||
static struct qtmiss_ops qtfs_miss_handles[] = {
|
||||
{QTFS_REQ_NULL, NULL, "null"},
|
||||
{QTFS_REQ_MOUNT, NULL, "mount"},
|
||||
{QTFS_REQ_OPEN, miss_open, "open"},
|
||||
};
|
||||
|
||||
int qtfs_missmsg_proc(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
struct qtreq *req = (struct qtreq *)pvar->vec_send.iov_base;
|
||||
struct qtreq *rsp = (struct qtreq *)pvar->vec_recv.iov_base;
|
||||
int ret;
|
||||
qtfs_err("qtfs miss message proc req type:%u rsp type:%u.", req->type, rsp->type);
|
||||
if (rsp->type > QTFS_REQ_OPEN) {
|
||||
qtfs_err("qtfs miss message proc failed, type:%u invalid, req type:%u.", rsp->type, req->type);
|
||||
return -EINVAL;
|
||||
}
|
||||
if (qtfs_miss_handles[rsp->type].misshandle == NULL) {
|
||||
qtfs_err("qtfs miss message proc not support:%u, req type:%u.", rsp->type, req->type);
|
||||
return -ESRCH;
|
||||
}
|
||||
ret = qtfs_miss_handles[rsp->type].misshandle(rsp);
|
||||
if (ret != QTFS_OK) {
|
||||
qtfs_err("qtfs miss message proc failed, req type:%u rsp type:%u.", req->type, rsp->type);
|
||||
}
|
||||
return ret;
|
||||
}
|
43
qtfs/qtfs/ops.h
Normal file
43
qtfs/qtfs/ops.h
Normal file
@ -0,0 +1,43 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_OPS_H__
|
||||
#define __QTFS_OPS_H__
|
||||
|
||||
#include <linux/fs.h>
|
||||
#include <linux/version.h>
|
||||
|
||||
#include "qtfs-mod.h"
|
||||
|
||||
extern struct inode_operations qtfs_proc_inode_ops;
|
||||
extern struct file_operations qtfs_proc_file_ops;
|
||||
extern struct inode_operations qtfs_proc_sym_ops;
|
||||
extern struct file_operations qtfsfifo_ops;
|
||||
|
||||
enum qtfs_type qtfs_get_type(char *str);
|
||||
bool is_sb_proc(struct super_block *sb);
|
||||
|
||||
struct inode *qtfs_iget(struct super_block *sb, struct inode_info *ii);
|
||||
const char *qtfs_getlink(struct dentry *dentry,
|
||||
struct inode *inode, struct delayed_call *done);
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(6, 4, 0))
|
||||
int qtfs_getattr(struct mnt_idmap *idmap, const struct path *, struct kstat *, u32, unsigned int);
|
||||
#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
int qtfs_getattr(struct user_namespace *mnt_userns, const struct path *, struct kstat *, u32, unsigned int);
|
||||
#else
|
||||
int qtfs_getattr(const struct path *, struct kstat *, u32, unsigned int);
|
||||
#endif
|
||||
struct dentry * qtfs_lookup(struct inode *, struct dentry *, unsigned int);
|
||||
__poll_t qtfsfifo_poll(struct file *filp, poll_table *wait);
|
||||
|
||||
#endif
|
310
qtfs/qtfs/proc.c
Normal file
310
qtfs/qtfs/proc.c
Normal file
@ -0,0 +1,310 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/time.h>
|
||||
#include <linux/fs_struct.h>
|
||||
#include <linux/version.h>
|
||||
#include <linux/sched/task.h>
|
||||
|
||||
#include "conn.h"
|
||||
#include "qtfs-mod.h"
|
||||
#include "req.h"
|
||||
#include "log.h"
|
||||
#include "ops.h"
|
||||
#include "symbol_wrapper.h"
|
||||
|
||||
struct dentry *qtfs_proc_lookup(struct inode *parent_inode, struct dentry *child_dentry, unsigned int flags);
|
||||
const char *qtfs_proc_getlink(struct dentry *dentry, struct inode *inode, struct delayed_call *done);
|
||||
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(6, 4, 0))
|
||||
int qtfs_proc_getattr(struct mnt_idmap *idmap, const struct path *path, struct kstat *stat, u32 req_mask, unsigned int flags);
|
||||
#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
int qtfs_proc_getattr(struct user_namespace *mnt_userns, const struct path *path, struct kstat *stat, u32 req_mask, unsigned int flags);
|
||||
#else
|
||||
int qtfs_proc_getattr(const struct path *path, struct kstat *stat, u32 req_mask, unsigned int flags);
|
||||
#endif
|
||||
|
||||
enum qtfs_type qtfs_get_type(char *str)
|
||||
{
|
||||
if (str && !strcmp(str, "proc"))
|
||||
return QTFS_PROC;
|
||||
return QTFS_NORMAL;
|
||||
}
|
||||
|
||||
bool is_sb_proc(struct super_block *sb)
|
||||
{
|
||||
struct qtfs_fs_info *qfi = NULL;
|
||||
|
||||
if (!sb || !sb->s_fs_info)
|
||||
return false;
|
||||
|
||||
qfi = sb->s_fs_info;
|
||||
return qfi->type == QTFS_PROC;
|
||||
}
|
||||
|
||||
const char *match_str[] = {
|
||||
// k8s isula and docker
|
||||
"bash",
|
||||
"isula",
|
||||
"isulad",
|
||||
"isulad-real",
|
||||
"kubelet",
|
||||
"kubelet-real",
|
||||
"dockerd",
|
||||
"dockerd-real",
|
||||
"containerd",
|
||||
"containerd-real",
|
||||
// Virtualization scene
|
||||
"libvirtd",
|
||||
"virsh",
|
||||
"rpc-worker",
|
||||
"rexec",
|
||||
};
|
||||
|
||||
int is_local_process(const char *path)
|
||||
{
|
||||
int pid = -1;
|
||||
char cmdline[TASK_COMM_LEN];
|
||||
char *pos = NULL;
|
||||
struct task_struct *t = NULL;
|
||||
int i = 0;
|
||||
|
||||
sscanf(path, "/proc/%d", &pid);
|
||||
if (pid <= 0)
|
||||
return -1;
|
||||
|
||||
t = qtfs_kern_syms.find_get_task_by_vpid((pid_t)pid);
|
||||
if (!t) {
|
||||
qtfs_info("[is_local_process] Failed to get task_struct from pid(%d)", pid);
|
||||
return -1;
|
||||
}
|
||||
get_task_comm(cmdline, t);
|
||||
put_task_struct(t);
|
||||
|
||||
pos = strrchr(cmdline, '/');
|
||||
if (!pos) {
|
||||
pos = cmdline;
|
||||
} else {
|
||||
pos++;
|
||||
}
|
||||
|
||||
for (i = 0; i < sizeof(match_str)/sizeof(char *); i++) {
|
||||
if (!strncmp(pos, match_str[i], NAME_MAX)) {
|
||||
qtfs_debug("[is_local_process] cmdline: %s is local process %d\n", cmdline, pid);
|
||||
return pid;
|
||||
}
|
||||
}
|
||||
|
||||
qtfs_debug("[is_local_process] cmdline: %s is not local process", cmdline);
|
||||
return -1;
|
||||
}
|
||||
|
||||
struct inode_operations qtfs_proc_inode_ops = {
|
||||
.lookup = qtfs_proc_lookup,
|
||||
.getattr = qtfs_proc_getattr,
|
||||
};
|
||||
|
||||
struct inode_operations qtfs_proc_sym_ops = {
|
||||
.get_link = qtfs_proc_getlink,
|
||||
.getattr = qtfs_proc_getattr,
|
||||
};
|
||||
|
||||
struct dentry *qtfs_proc_lookup(struct inode *parent_inode, struct dentry *child_dentry, unsigned int flags)
|
||||
{
|
||||
char *cpath = NULL, *tmp = NULL;
|
||||
struct path spath;
|
||||
struct dentry *d = NULL;
|
||||
struct inode_info ii;
|
||||
struct inode *inode = NULL;
|
||||
int ret = 0;
|
||||
int pid = -1;
|
||||
|
||||
cpath = kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
tmp = kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
if (!tmp || !cpath) {
|
||||
qtfs_err("%s: failed to alloc memory", __func__);
|
||||
goto remote;
|
||||
}
|
||||
memset(cpath, 0, MAX_PATH_LEN);
|
||||
memset(tmp, 0, MAX_PATH_LEN);
|
||||
|
||||
if (qtfs_fullname(cpath, child_dentry, MAX_PATH_LEN) < 0) {
|
||||
qtfs_err("%s: failed to get fullname", __func__);
|
||||
goto remote;
|
||||
}
|
||||
|
||||
pid = is_local_process(cpath);
|
||||
if (pid > 0) {
|
||||
sscanf(cpath, "/proc/%s", tmp);
|
||||
memset(cpath, 0, MAX_PATH_LEN);
|
||||
sprintf(cpath, "/local_proc/%s", tmp);
|
||||
qtfs_debug("[%s]: get path from local: %s\n", __func__, cpath);
|
||||
ret = kern_path(cpath, 0, &spath);
|
||||
if(ret) {
|
||||
qtfs_err("[%s]: kern_path(%s) failed: %d\n", __func__, cpath, ret);
|
||||
goto remote;
|
||||
}
|
||||
|
||||
ii.mode = spath.dentry->d_inode->i_mode;
|
||||
ii.mode = (ii.mode & ~(S_IFMT)) | S_IFLNK;
|
||||
ii.i_size = spath.dentry->d_inode->i_size;
|
||||
ii.i_ino = spath.dentry->d_inode->i_ino;
|
||||
ii.atime = spath.dentry->d_inode->i_atime;
|
||||
ii.mtime = spath.dentry->d_inode->i_mtime;
|
||||
ii.ctime = spath.dentry->d_inode->i_ctime;
|
||||
path_put(&spath);
|
||||
|
||||
kfree(tmp);
|
||||
tmp = NULL;
|
||||
inode = qtfs_iget(parent_inode->i_sb, &ii);
|
||||
if (inode == NULL) {
|
||||
qtfs_err("%s: failed to get inode for %s", __func__, cpath);
|
||||
kfree(cpath);
|
||||
return ERR_PTR(-ENOMEM);
|
||||
}
|
||||
d = d_splice_alias(inode, child_dentry);
|
||||
kfree(cpath);
|
||||
return d;
|
||||
}
|
||||
|
||||
remote:
|
||||
if (cpath)
|
||||
kfree(cpath);
|
||||
if (tmp)
|
||||
kfree(tmp);
|
||||
return qtfs_lookup(parent_inode, child_dentry, flags);
|
||||
}
|
||||
|
||||
const char *qtfs_proc_getlink(struct dentry *dentry,
|
||||
struct inode *inode, struct delayed_call *done)
|
||||
{
|
||||
char *link = NULL, *path = NULL, *tmp = NULL;
|
||||
int pid = -1;
|
||||
|
||||
link = kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
path = kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
tmp = kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
if (!link || !tmp || !path) {
|
||||
qtfs_err("[%s]: failed to alloc memory", __func__);
|
||||
goto link_remote;
|
||||
}
|
||||
memset(link, 0, MAX_PATH_LEN);
|
||||
memset(path, 0, MAX_PATH_LEN);
|
||||
memset(tmp, 0, MAX_PATH_LEN);
|
||||
|
||||
if (qtfs_fullname(path, dentry, MAX_PATH_LEN) < 0) {
|
||||
qtfs_info("[%s]: get path failed", __func__);
|
||||
goto link_remote;
|
||||
}
|
||||
|
||||
if (!strncmp(path, "/proc/self", 11)) {
|
||||
sprintf(link, "/local_proc/%d", (int)current->pid);
|
||||
qtfs_info("[%s] success: %s getlink: %s", __func__, path, link);
|
||||
goto link_local;
|
||||
}
|
||||
|
||||
if (!strcmp(path, "/proc/mounts")) {
|
||||
sprintf(link, "/proc/1/mounts");
|
||||
qtfs_info("[%s] success: %s getlink /proc/1/mounts", __func__, path);
|
||||
goto link_local;
|
||||
}
|
||||
|
||||
pid = is_local_process(path);
|
||||
if (pid > 0) {
|
||||
sscanf(path, "/proc/%s", tmp);
|
||||
sprintf(link, "/local_proc/%s", tmp);
|
||||
qtfs_info("[%s] success: %s getlink: %s", __func__, path, link);
|
||||
goto link_local;
|
||||
}
|
||||
|
||||
link_remote:
|
||||
if (link)
|
||||
kfree(link);
|
||||
if (tmp)
|
||||
kfree(tmp);
|
||||
if (path)
|
||||
kfree(path);
|
||||
return qtfs_getlink(dentry, inode, done);
|
||||
link_local:
|
||||
kfree(tmp);
|
||||
kfree(path);
|
||||
set_delayed_call(done, kfree_link, link);
|
||||
return link;
|
||||
}
|
||||
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(6, 4, 0))
|
||||
int qtfs_proc_getattr(struct mnt_idmap *idmap, const struct path *path, struct kstat *stat, u32 req_mask, unsigned int flags)
|
||||
#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
int qtfs_proc_getattr(struct user_namespace *mnt_userns, const struct path *path, struct kstat *stat, u32 req_mask, unsigned int flags)
|
||||
#else
|
||||
int qtfs_proc_getattr(const struct path *path, struct kstat *stat, u32 req_mask, unsigned int flags)
|
||||
#endif
|
||||
{
|
||||
char *cpath = NULL, *tmp = NULL, *local_path = NULL;
|
||||
struct path spath;
|
||||
int ret = 0;
|
||||
int pid = -1;
|
||||
|
||||
cpath = kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
tmp = kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
local_path = kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
if (!cpath || !tmp || !local_path) {
|
||||
qtfs_err("[%s]: failed to alloc memory", __func__);
|
||||
goto remote;
|
||||
}
|
||||
memset(cpath, 0, MAX_PATH_LEN);
|
||||
memset(tmp, 0, MAX_PATH_LEN);
|
||||
memset(local_path, 0, MAX_PATH_LEN);
|
||||
|
||||
if (qtfs_fullname(cpath, path->dentry, MAX_PATH_LEN) < 0) {
|
||||
qtfs_err("%s: failed to get fullname", __func__);
|
||||
goto remote;
|
||||
}
|
||||
|
||||
pid = is_local_process(cpath);
|
||||
if (pid > 0) {
|
||||
sscanf(cpath, "/proc/%s", tmp);
|
||||
sprintf(local_path, "/local_proc/%s", tmp);
|
||||
ret = kern_path(local_path, 0, &spath);
|
||||
if (ret) {
|
||||
qtfs_err("[%s]: kern_path(%s) failed: %d", __func__, local_path, ret);
|
||||
goto remote;
|
||||
}
|
||||
|
||||
ret = vfs_getattr(&spath, stat, req_mask, flags);
|
||||
path_put(&spath);
|
||||
if (ret) {
|
||||
qtfs_err("[%s]: vfs_getattr %s failed: %d", __func__, local_path, ret);
|
||||
goto remote;
|
||||
}
|
||||
qtfs_debug("[%s]: %s success", __func__, local_path);
|
||||
stat->mode = (stat->mode & ~(S_IFMT)) | S_IFLNK;
|
||||
kfree(cpath);
|
||||
kfree(tmp);
|
||||
kfree(local_path);
|
||||
return 0;
|
||||
}
|
||||
|
||||
remote:
|
||||
if (cpath)
|
||||
kfree(cpath);
|
||||
if (tmp)
|
||||
kfree(tmp);
|
||||
if (local_path)
|
||||
kfree(local_path);
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
return qtfs_getattr(NULL, path, stat, req_mask, flags);
|
||||
#else
|
||||
return qtfs_getattr(path, stat, req_mask, flags);
|
||||
#endif
|
||||
}
|
316
qtfs/qtfs/qtfs-mod.c
Normal file
316
qtfs/qtfs/qtfs-mod.c
Normal file
@ -0,0 +1,316 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/ktime.h>
|
||||
#include <linux/kthread.h>
|
||||
#include <linux/version.h>
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
#include <linux/sched/task.h>
|
||||
#endif
|
||||
#include "conn.h"
|
||||
|
||||
#include "qtfs-mod.h"
|
||||
#include "syscall.h"
|
||||
#include "symbol_wrapper.h"
|
||||
|
||||
static struct file_system_type qtfs_fs_type = {
|
||||
.owner = THIS_MODULE,
|
||||
.name = QTFS_FSTYPE_NAME,
|
||||
.mount = qtfs_fs_mount,
|
||||
.kill_sb = qtfs_kill_sb,
|
||||
};
|
||||
MODULE_ALIAS_FS("qtfs");
|
||||
struct kmem_cache *qtfs_inode_priv_cache;
|
||||
struct task_struct *g_qtfs_epoll_thread = NULL;
|
||||
|
||||
/*
|
||||
* 转发框架层:
|
||||
* 1. 调用者先在pvar里预留框架头后,填好自己的私有发送数据。
|
||||
2. 框架负责填充框架头,将整体消息发到对端。
|
||||
3. 等待对端执行完回复。
|
||||
4. 将接收buf的私有数据段首指针返回给调用者,完成文件操作层的通信。
|
||||
*/
|
||||
void *qtfs_remote_run(struct qtfs_conn_var_s *pvar, unsigned int type, unsigned int len)
|
||||
{
|
||||
int ret;
|
||||
unsigned long retrytimes = 0;
|
||||
struct qtreq *req = (struct qtreq *)pvar->vec_send.iov_base;
|
||||
struct qtreq *rsp = (struct qtreq *)pvar->vec_recv.iov_base;
|
||||
if (req == NULL || type >= QTFS_REQ_INV) {
|
||||
qtfs_err("qtfs remote run failed, req is NULL type:%u.\n", type);
|
||||
return NULL;
|
||||
}
|
||||
pvar->seq_num++;
|
||||
req->type = type;
|
||||
req->len = len;
|
||||
req->err = 0;
|
||||
req->seq_num = pvar->seq_num;
|
||||
|
||||
pvar->conn_ops->conn_recv_buff_drop(&pvar->conn_var);
|
||||
// 调用qtfs_remote_run之前,调用者应该先把消息在iov_base里面封装好
|
||||
// 如果不是socket通信,则是在其他通信模式定义的buf里,消息协议统一
|
||||
// 都是struct qtreq *xx
|
||||
// 给server发一个消息
|
||||
pvar->vec_send.iov_len = QTFS_MSG_HEAD_LEN + len;
|
||||
ret = qtfs_conn_send(pvar);
|
||||
if (ret <= 0) {
|
||||
qtfs_err("qtfs remote run send failed, ret:%d pvar sendlen:%lu.", ret, pvar->vec_send.iov_len);
|
||||
qtinfo_senderrinc(req->type);
|
||||
return NULL;
|
||||
}
|
||||
qtinfo_sendinc(type);
|
||||
|
||||
// wait for response
|
||||
retry:
|
||||
ret = qtfs_conn_recv_block(pvar);
|
||||
if (ret == -EAGAIN)
|
||||
goto retry;
|
||||
if (ret > 0 && req->seq_num != rsp->seq_num) {
|
||||
qtinfo_cntinc(QTINF_SEQ_ERR);
|
||||
qtfs_debug("qtfs remote run recv msg mismatch type:%d, ret:%d pvaridx:%d req:%lu rsp:%lu.",
|
||||
req->type, ret, pvar->cur_threadidx, req->seq_num, rsp->seq_num);
|
||||
qtinfo_recvinc(rsp->type);
|
||||
if (pvar->miss_proc == 0) {
|
||||
pvar->miss_proc = 1;
|
||||
qtfs_missmsg_proc(pvar);
|
||||
pvar->miss_proc = 0;
|
||||
}
|
||||
goto retry;
|
||||
}
|
||||
if (ret == -ERESTARTSYS) {
|
||||
if (retrytimes == 0) {
|
||||
qtinfo_cntinc(QTINF_RESTART_SYS);
|
||||
qtinfo_recverrinc(req->type);
|
||||
}
|
||||
retrytimes++;
|
||||
msleep(1);
|
||||
goto retry;
|
||||
}
|
||||
if (ret < 0) {
|
||||
qtfs_err("qtfs remote run error, ret:%d.", ret);
|
||||
qtinfo_recverrinc(req->type);
|
||||
return NULL;
|
||||
}
|
||||
if (retrytimes > 0)
|
||||
qtfs_debug("qtfs remote run retry times:%lu.", retrytimes);
|
||||
qtinfo_recvinc(rsp->type);
|
||||
|
||||
if (rsp->err == QTFS_ERR) {
|
||||
qtfs_err("qtfs remote run error, req errcode:%d type:%u len:%lu\n", req->err, req->type, req->len);
|
||||
return NULL;
|
||||
}
|
||||
return pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_RECV);
|
||||
}
|
||||
|
||||
static int qtfs_epoll_thread(void *data)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = NULL;
|
||||
struct qtreq_epollevt *req;
|
||||
struct qtrsp_epollevt *rsp;
|
||||
struct qtreq *head;
|
||||
int ret;
|
||||
struct inode *inode;
|
||||
struct file *file;
|
||||
int i;
|
||||
|
||||
connecting:
|
||||
while (qtfs_mod_exiting == false) {
|
||||
pvar = qtfs_epoll_establish_conn();
|
||||
if (pvar != NULL)
|
||||
break;
|
||||
msleep(500);
|
||||
}
|
||||
if (pvar == NULL) {
|
||||
goto end;
|
||||
}
|
||||
qtfs_info("qtfs epoll thread establish a new connection.");
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_RECV);
|
||||
rsp = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
|
||||
// init ack head only once
|
||||
do {
|
||||
head = pvar->vec_send.iov_base;
|
||||
pvar->vec_send.iov_len = QTFS_MSG_HEAD_LEN + sizeof(struct qtrsp_epollevt);
|
||||
head->type = QTFS_REQ_EPOLL_EVENT;
|
||||
head->len = sizeof(struct qtrsp_epollevt);
|
||||
rsp->ret = QTFS_OK;
|
||||
} while (0);
|
||||
|
||||
while (!kthread_should_stop()) {
|
||||
ret = qtfs_conn_recv(pvar);
|
||||
if (ret == -EPIPE || pvar->conn_ops->conn_connected(&pvar->conn_var) == false)
|
||||
goto connecting;
|
||||
if (ret < 0 || req->event_nums <= 0 || req->event_nums >= QTFS_MAX_EPEVENTS_NUM) {
|
||||
continue;
|
||||
}
|
||||
qtfs_debug("epoll thread recv %d events.", req->event_nums);
|
||||
for (i = 0; i < req->event_nums; i++) {
|
||||
// events[i].data is *file ptr
|
||||
file = (struct file *)req->events[i].data;
|
||||
if (IS_ERR_OR_NULL(file)) {
|
||||
qtfs_err("epoll thread event file invalid!");
|
||||
continue;
|
||||
}
|
||||
inode = file->f_inode;
|
||||
// 暂时只支持fifo文件的epoll
|
||||
if (inode == NULL || !qtfs_support_epoll(inode->i_mode)) {
|
||||
qtfs_err("epoll thread event file not a fifo.");
|
||||
continue;
|
||||
}
|
||||
do {
|
||||
struct qtfs_inode_priv *priv = inode->i_private;
|
||||
__poll_t key;
|
||||
if (req->events[i].events & EPOLLHUP)
|
||||
key = EPOLLHUP;
|
||||
else
|
||||
key = EPOLLIN | EPOLLRDNORM;
|
||||
if (priv == NULL) {
|
||||
qtfs_err("epoll epoll wake up file error, inode priv is invalid.");
|
||||
WARN_ON(1);
|
||||
} else {
|
||||
wake_up_interruptible_sync_poll(&priv->readq, key);
|
||||
}
|
||||
} while (0);
|
||||
}
|
||||
ret = qtfs_conn_send(pvar);
|
||||
if (ret < 0)
|
||||
qtfs_err("conn send failed, ret:%d\n", ret);
|
||||
}
|
||||
end:
|
||||
g_qtfs_epoll_thread = NULL;
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct file_operations qtfs_misc_fops = {
|
||||
.owner=THIS_MODULE,
|
||||
.unlocked_ioctl = qtfs_misc_ioctl,
|
||||
};
|
||||
|
||||
static int __init qtfs_init(void)
|
||||
{
|
||||
int ret;
|
||||
qtfs_log_init(qtfs_log_level, sizeof(qtfs_log_level));
|
||||
|
||||
if (qtfs_kallsyms_hack_init() != 0)
|
||||
return -1;
|
||||
|
||||
ret = register_filesystem(&qtfs_fs_type);
|
||||
if (ret != 0) {
|
||||
qtfs_err("QTFS file system register failed, ret:%d.\n", ret);
|
||||
return -1;
|
||||
}
|
||||
qtfs_inode_priv_cache = kmem_cache_create("qtfs_inode_priv",
|
||||
sizeof(struct qtfs_inode_priv),
|
||||
0,
|
||||
(SLAB_RECLAIM_ACCOUNT | SLAB_MEM_SPREAD),
|
||||
NULL);
|
||||
if (!qtfs_inode_priv_cache) {
|
||||
qtfs_err("qtfs inode priv cache create failed.\n");
|
||||
ret = -ENOMEM;
|
||||
goto cache_create_err;
|
||||
}
|
||||
if (qtfs_conn_param_init() < 0) {
|
||||
ret = -ENOMEM;
|
||||
goto conn_init_err;
|
||||
}
|
||||
qtfs_whitelist_initset();
|
||||
qtfs_conn_param_init();
|
||||
g_qtfs_epoll_thread = kthread_run(qtfs_epoll_thread, NULL, "qtfs_epoll");
|
||||
if (IS_ERR_OR_NULL(g_qtfs_epoll_thread)) {
|
||||
qtfs_err("qtfs epoll thread run failed.\n");
|
||||
ret = QTFS_PTR_ERR(g_qtfs_epoll_thread);
|
||||
goto epoll_thread_err;
|
||||
}
|
||||
qtfs_diag_info = (struct qtinfo *)kmalloc(sizeof(struct qtinfo), GFP_KERNEL);
|
||||
if (qtfs_diag_info == NULL) {
|
||||
qtfs_err("kmalloc qtfs diag info failed.");
|
||||
ret = -ENOMEM;
|
||||
goto diag_malloc_err;
|
||||
} else {
|
||||
memset(qtfs_diag_info, 0, sizeof(struct qtinfo));
|
||||
}
|
||||
ret = qtfs_misc_register();
|
||||
if (ret) {
|
||||
goto misc_register_err;
|
||||
}
|
||||
ret = qtfs_syscall_replace_start();
|
||||
if (ret) {
|
||||
goto syscall_replace_err;
|
||||
}
|
||||
ret = qtfs_syscall_init();
|
||||
if (ret) {
|
||||
goto syscall_init_err;
|
||||
}
|
||||
ret = qtfs_utils_register();
|
||||
if (ret) {
|
||||
goto utils_register_err;
|
||||
}
|
||||
qtfs_info("QTFS file system register success!\n");
|
||||
return 0;
|
||||
utils_register_err:
|
||||
qtfs_syscall_fini();
|
||||
syscall_init_err:
|
||||
qtfs_syscall_replace_stop();
|
||||
syscall_replace_err:
|
||||
qtfs_misc_destroy();
|
||||
misc_register_err:
|
||||
kfree(qtfs_diag_info);
|
||||
diag_malloc_err:
|
||||
kthread_stop(g_qtfs_epoll_thread);
|
||||
epoll_thread_err:
|
||||
qtfs_conn_param_fini();
|
||||
conn_init_err:
|
||||
kmem_cache_destroy(qtfs_inode_priv_cache);
|
||||
cache_create_err:
|
||||
unregister_filesystem(&qtfs_fs_type);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void __exit qtfs_exit(void)
|
||||
{
|
||||
int ret;
|
||||
qtfs_mod_exiting = true;
|
||||
|
||||
if (g_qtfs_epoll_thread) {
|
||||
kthread_stop(g_qtfs_epoll_thread);
|
||||
}
|
||||
|
||||
qtfs_conn_param_fini();
|
||||
qtfs_misc_destroy();
|
||||
if (qtfs_epoll_var != NULL) {
|
||||
qtfs_epoll_cut_conn(qtfs_epoll_var);
|
||||
qtfs_epoll_var->conn_ops->conn_var_fini(qtfs_epoll_var);
|
||||
kfree(qtfs_epoll_var);
|
||||
qtfs_epoll_var = NULL;
|
||||
}
|
||||
qtfs_whitelist_exit();
|
||||
|
||||
kfree(qtfs_diag_info);
|
||||
qtfs_diag_info = NULL;
|
||||
qtfs_syscall_fini();
|
||||
|
||||
ret = unregister_filesystem(&qtfs_fs_type);
|
||||
if (ret != 0) {
|
||||
qtfs_err("QTFS file system unregister failed, ret:%d.\n", ret);
|
||||
}
|
||||
qtfs_utils_destroy();
|
||||
kmem_cache_destroy(qtfs_inode_priv_cache);
|
||||
qtfs_syscall_replace_stop();
|
||||
qtfs_info("QTFS file system unregister success!\n");
|
||||
return;
|
||||
}
|
||||
|
||||
module_init(qtfs_init);
|
||||
module_exit(qtfs_exit);
|
||||
MODULE_AUTHOR("liqiang64@huawei.com");
|
||||
MODULE_LICENSE("GPL");
|
196
qtfs/qtfs/qtfs-mod.h
Normal file
196
qtfs/qtfs/qtfs-mod.h
Normal file
@ -0,0 +1,196 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_INCLUDE_H__
|
||||
#define __QTFS_INCLUDE_H__
|
||||
|
||||
#include <linux/buffer_head.h>
|
||||
#include <linux/fs.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/namei.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/cdev.h>
|
||||
#include <linux/sched.h>
|
||||
#include <linux/dcache.h>
|
||||
#include <linux/uaccess.h>
|
||||
#include <linux/parser.h>
|
||||
#include <linux/random.h>
|
||||
#include <linux/errno.h>
|
||||
#include <asm/current.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/time.h>
|
||||
#include <linux/version.h>
|
||||
#include <linux/dirent.h>
|
||||
|
||||
#include "comm.h"
|
||||
#include "log.h"
|
||||
#include "req.h"
|
||||
|
||||
|
||||
#define QTFS_MAXLEN 8
|
||||
#define QTFS_MAX_FILES 32
|
||||
#define QTFS_MAX_BLOCKSIZE 512
|
||||
|
||||
#define QTFS_FSTYPE_NAME "qtfs"
|
||||
|
||||
extern struct kmem_cache *qtfs_inode_priv_cache;
|
||||
|
||||
struct private_data {
|
||||
int fd;
|
||||
};
|
||||
|
||||
struct qtfs_inode_priv {
|
||||
unsigned int files;
|
||||
wait_queue_head_t readq;
|
||||
wait_queue_head_t writeq;
|
||||
};
|
||||
|
||||
enum {
|
||||
QTFS_ROOT_INO = 1,
|
||||
QTFS_IPC_INIT_INO = 0xEFFFFFFFU,
|
||||
QTFS_UTS_INIT_INO = 0xEFFFFFFEU,
|
||||
QTFS_USER_INIT_INO = 0xEFFFFFFDU,
|
||||
QTFS_PID_INIT_INO = 0xEFFFFFFCU,
|
||||
QTFS_CGROUP_INIT_INO = 0xEFFFFFFBU,
|
||||
QTFS_TIME_INIT_INO = 0xEFFFFFFAU,
|
||||
QTFS_IMA_INIT_INO = 0xEFFFFFF9U,
|
||||
};
|
||||
|
||||
struct qtfs_inode {
|
||||
mode_t mode;
|
||||
uint64_t i_no;
|
||||
uint64_t d_no;
|
||||
char *peer_path;
|
||||
union {
|
||||
uint64_t file_size;
|
||||
uint64_t dir_childrens;
|
||||
};
|
||||
struct list_head entry;
|
||||
};
|
||||
|
||||
struct qtfs_fs_info {
|
||||
char peer_path[NAME_MAX];
|
||||
char *mnt_path;
|
||||
|
||||
enum qtfs_type type;
|
||||
};
|
||||
|
||||
struct qtfs_dir_entry {
|
||||
struct list_head node;
|
||||
char filename[NAME_MAX];
|
||||
struct qtfs_inode *priv;
|
||||
};
|
||||
|
||||
struct qtfs_file_blk {
|
||||
uint8_t busy;
|
||||
mode_t mode;
|
||||
uint8_t idx;
|
||||
|
||||
union {
|
||||
uint8_t file_size;
|
||||
uint8_t dir_children;
|
||||
};
|
||||
char data[0];
|
||||
};
|
||||
|
||||
struct qtmiss_ops {
|
||||
int type;
|
||||
// return int is output len.
|
||||
int (*misshandle) (struct qtreq *);
|
||||
char str[32];
|
||||
};
|
||||
|
||||
static inline int qtfs_fullname(char *fullname, struct dentry *d, size_t buflen)
|
||||
{
|
||||
struct qtfs_fs_info *fsinfo = NULL;
|
||||
int len = 0;
|
||||
char *name = NULL;
|
||||
char *ret = NULL;
|
||||
|
||||
if (!d) {
|
||||
qtfs_info("%s: get dentry fullname NULL\n", __func__);
|
||||
return -1;
|
||||
}
|
||||
if (buflen < MAX_PATH_LEN) {
|
||||
qtfs_err("%s: failed to get fullname dure to small buflen:%lu\n", __func__, buflen);
|
||||
return -1;
|
||||
}
|
||||
|
||||
name = __getname();
|
||||
if (!name) {
|
||||
return -1;
|
||||
}
|
||||
ret = dentry_path_raw(d, name, MAX_PATH_LEN);
|
||||
if (IS_ERR_OR_NULL(ret)) {
|
||||
qtfs_err("qtfs fullname failed:%ld\n", QTFS_PTR_ERR(ret));
|
||||
__putname(name);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (d && d->d_sb && d->d_sb->s_fs_info) {
|
||||
fsinfo = d->d_sb->s_fs_info;
|
||||
} else {
|
||||
qtfs_err("%s: failed to get private fs_info\n", __func__);
|
||||
__putname(name);
|
||||
return -1;
|
||||
}
|
||||
if (strcmp(fsinfo->peer_path, "/")) {
|
||||
/* if peer_path is not root '/' */
|
||||
len = strlcpy(fullname, fsinfo->peer_path, MAX_PATH_LEN);
|
||||
}
|
||||
if (len + strlen(ret) >= MAX_PATH_LEN - 1) {
|
||||
qtfs_err("qtfs fullname may reach max len:%d reallen:%ld path:%s", len, strlen(fullname), fullname);
|
||||
__putname(name);
|
||||
return -1;
|
||||
}
|
||||
len += strlcpy(&fullname[len], ret, MAX_PATH_LEN - len);
|
||||
if (strcmp(fullname, "/")) {
|
||||
if (fullname[strlen(fullname) - 1] == '/')
|
||||
fullname[strlen(fullname) - 1] = '\0';
|
||||
}
|
||||
__putname(name);
|
||||
return len;
|
||||
}
|
||||
|
||||
#define QTFS_FULLNAME(fullname, d, buflen) \
|
||||
if (qtfs_fullname(fullname, d, buflen)<0) { \
|
||||
qtfs_err("qtfs fullname failed\n"); \
|
||||
qtfs_conn_put_param(pvar); \
|
||||
return -EINVAL; \
|
||||
}
|
||||
|
||||
extern const struct xattr_handler qtfs_xattr_user_handler;
|
||||
extern const struct xattr_handler qtfs_xattr_trusted_handler;
|
||||
extern const struct xattr_handler qtfs_xattr_security_handler;
|
||||
extern const struct xattr_handler qtfs_xattr_hurd_handler;
|
||||
extern struct qtinfo *qtfs_diag_info;
|
||||
extern int qtfs_mod_exiting;
|
||||
|
||||
void qtfs_kill_sb(struct super_block *sb);
|
||||
struct dentry *qtfs_fs_mount(struct file_system_type *fs_type,
|
||||
int flags, const char *dev_name,
|
||||
void *data);
|
||||
void *qtfs_remote_run(struct qtfs_conn_var_s *pvar, unsigned int type, unsigned int len);
|
||||
int qtfs_misc_register(void);
|
||||
void qtfs_misc_destroy(void);
|
||||
void qtfs_whitelist_exit(void);
|
||||
long qtfs_misc_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
int qtfs_missmsg_proc(struct qtfs_conn_var_s *pvar);
|
||||
int qtfs_utils_register(void);
|
||||
void qtfs_utils_destroy(void);
|
||||
void qtfs_whitelist_clearall(void);
|
||||
void qtfs_whitelist_initset(void);
|
||||
|
||||
#endif
|
||||
|
1668
qtfs/qtfs/sb.c
Normal file
1668
qtfs/qtfs/sb.c
Normal file
File diff suppressed because it is too large
Load Diff
434
qtfs/qtfs/syscall.c
Normal file
434
qtfs/qtfs/syscall.c
Normal file
@ -0,0 +1,434 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/compiler_types.h>
|
||||
#include <linux/syscalls.h>
|
||||
#include <linux/trace_events.h>
|
||||
#include <linux/mount.h>
|
||||
#include <linux/file.h>
|
||||
#include <linux/eventpoll.h>
|
||||
#include <linux/atomic.h>
|
||||
|
||||
#include "conn.h"
|
||||
#include "qtfs-mod.h"
|
||||
#include "symbol_wrapper.h"
|
||||
|
||||
static long qtfs_remote_mount(char __user *dev_name, char __user *dir_name, char __user *type,
|
||||
unsigned long flags, void __user *data);
|
||||
static int qtfs_remote_umount(char __user *name, int flags);
|
||||
|
||||
#ifdef BEFORE_KVER_5_6
|
||||
static inline int ep_op_has_event(int op)
|
||||
{
|
||||
return op != EPOLL_CTL_DEL;
|
||||
}
|
||||
#endif
|
||||
|
||||
static char *qtfs_copy_mount_string(const void __user *data)
|
||||
{
|
||||
return data ? strndup_user(data, PATH_MAX) : NULL;
|
||||
}
|
||||
|
||||
static inline int qtfs_fstype_judgment(char __user *dir)
|
||||
{
|
||||
struct path path;
|
||||
int ret;
|
||||
|
||||
ret = user_path_at(AT_FDCWD, dir, LOOKUP_FOLLOW, &path);
|
||||
if (ret)
|
||||
return 0;
|
||||
|
||||
if (path.mnt && path.mnt->mnt_sb &&
|
||||
path.mnt->mnt_sb->s_type && path.mnt->mnt_sb->s_type->name &&
|
||||
strcmp(path.mnt->mnt_sb->s_type->name, QTFS_FSTYPE_NAME) == 0) {
|
||||
qtfs_info("qtfs fstype judge <%s> is qtfs.\n", path.dentry->d_iname);
|
||||
path_put(&path);
|
||||
return 1;
|
||||
}
|
||||
path_put(&path);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* if this dir is root node of qtfs */
|
||||
static inline int qtfs_root_judgment(char __user *dir)
|
||||
{
|
||||
struct dentry *dentry;
|
||||
struct path path;
|
||||
int ret = 0;
|
||||
|
||||
ret = user_path_at(AT_FDCWD, dir, LOOKUP_FOLLOW, &path);
|
||||
if (ret)
|
||||
return 0;
|
||||
|
||||
dentry = path.dentry;
|
||||
if (dentry->d_parent == dentry)
|
||||
ret = 1;
|
||||
path_put(&path);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void do_epoll_ctl_remote(int op, struct epoll_event __user *event, struct file *file)
|
||||
{
|
||||
struct qtreq_epollctl *req;
|
||||
struct qtrsp_epollctl *rsp;
|
||||
struct qtfs_conn_var_s *pvar = qtfs_conn_get_param();
|
||||
struct private_data *priv = file->private_data;
|
||||
struct epoll_event tmp;
|
||||
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("qtfs do epoll ctl remote get pvar failed.");
|
||||
return;
|
||||
}
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
req->fd = priv->fd;
|
||||
req->op = op;
|
||||
if (ep_op_has_event(op) && copy_from_user(&tmp, event, sizeof(struct epoll_event))) {
|
||||
qtfs_err("qtfs do epoll ctl remote copy from user failed.");
|
||||
qtfs_conn_put_param(pvar);
|
||||
return;
|
||||
}
|
||||
req->event.events = tmp.events;
|
||||
req->event.data = (__u64)file;
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_EPOLL_CTL, sizeof(struct qtreq_epollctl));
|
||||
if (IS_ERR_OR_NULL(rsp) || rsp->ret == QTFS_ERR) {
|
||||
qtfs_err("qtfs do epoll ctl remote failed.");
|
||||
qtfs_conn_put_param(pvar);
|
||||
qtinfo_cntinc(QTINF_EPOLL_FDERR);
|
||||
return;
|
||||
}
|
||||
if (op == EPOLL_CTL_ADD) {
|
||||
qtinfo_cntinc(QTINF_EPOLL_ADDFDS);
|
||||
} else {
|
||||
qtinfo_cntinc(QTINF_EPOLL_DELFDS);
|
||||
}
|
||||
qtfs_info("qtfs do epoll ctl remote success, fd:%d.", req->fd);
|
||||
qtfs_conn_put_param(pvar);
|
||||
return;
|
||||
}
|
||||
|
||||
int qtfs_epoll_ctl_remote(int op, int fd, struct epoll_event __user * event)
|
||||
{
|
||||
struct fd f;
|
||||
struct file *file;
|
||||
struct private_data *priv;
|
||||
int ret = 0;
|
||||
f = fdget(fd);
|
||||
if (!f.file) {
|
||||
return -1;
|
||||
}
|
||||
file = f.file;
|
||||
if (strcmp(file->f_path.mnt->mnt_sb->s_type->name, QTFS_FSTYPE_NAME) != 0) {
|
||||
ret = 0;
|
||||
goto end;
|
||||
}
|
||||
if (!qtfs_support_epoll(file->f_inode->i_mode)) {
|
||||
char *fullname = (char *)kmalloc(MAX_PATH_LEN, GFP_KERNEL);
|
||||
if (!fullname) {
|
||||
ret = -1;
|
||||
goto end;
|
||||
}
|
||||
memset(fullname, 0, MAX_PATH_LEN);
|
||||
if (qtfs_fullname(fullname, file->f_path.dentry, MAX_PATH_LEN) < 0) {
|
||||
qtfs_err("qtfs fullname failed\n");
|
||||
kfree(fullname);
|
||||
ret = -1;
|
||||
goto end;
|
||||
}
|
||||
qtfs_info("qtfs remote epoll not support file:%s mode:%o.", fullname, file->f_inode->i_mode);
|
||||
kfree(fullname);
|
||||
ret = -1;
|
||||
goto end;
|
||||
}
|
||||
|
||||
priv = file->private_data;
|
||||
if (priv == NULL) {
|
||||
qtfs_err("epoll ctl remote failed, private data invalid.");
|
||||
ret = -1;
|
||||
goto end;
|
||||
}
|
||||
|
||||
qtfs_info("qtfs qtfs remote epoll file:%s mode:%x file can poll.",
|
||||
file->f_path.dentry->d_iname, file->f_inode->i_mode);
|
||||
do_epoll_ctl_remote(op, event, file);
|
||||
|
||||
end:
|
||||
fdput(f);
|
||||
return ret;
|
||||
}
|
||||
|
||||
__SYSCALL_DEFINEx(4, _qtfs_epoll_ctl, int, epfd, int, op, int, fd,
|
||||
struct epoll_event __user *, event)
|
||||
{
|
||||
int ret = -1;
|
||||
|
||||
ret = qtfs_epoll_ctl_remote(op, fd, event);
|
||||
if (!ret) {
|
||||
return qtfs_syscall_epoll_ctl(epfd, op, fd, event);
|
||||
} else {
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
__SYSCALL_DEFINEx(5, _qtfs_mount, char __user *, dev_name, char __user *, dir_name,
|
||||
char __user *, type, unsigned long, flags, void __user *, data)
|
||||
{
|
||||
int ret;
|
||||
char *kernel_type;
|
||||
char *kernel_dev;
|
||||
void *options = NULL;
|
||||
|
||||
// if both dev_name and dir_name are qtfs, it is a remote mount operator.
|
||||
kernel_type = qtfs_copy_mount_string(type);
|
||||
ret = PTR_ERR(kernel_type);
|
||||
if (IS_ERR(kernel_type))
|
||||
goto out_type;
|
||||
|
||||
kernel_dev = qtfs_copy_mount_string(dev_name);
|
||||
ret = PTR_ERR(kernel_dev);
|
||||
if (IS_ERR(kernel_dev))
|
||||
goto out_dev;
|
||||
|
||||
options = qtfs_copy_mount_string(data);
|
||||
ret = PTR_ERR(options);
|
||||
if (IS_ERR(options))
|
||||
goto out_data;
|
||||
|
||||
// if both dev_name and dir_name are qtfs, it is a remote mount operator,
|
||||
if (qtfs_fstype_judgment(dir_name) == 1) {
|
||||
ret = qtfs_remote_mount(kernel_dev, dir_name, kernel_type, flags, options);
|
||||
goto remote_mount;
|
||||
}
|
||||
|
||||
ret = qtfs_syscall_mount(dev_name, dir_name, type, flags, data);
|
||||
|
||||
remote_mount:
|
||||
kfree(options);
|
||||
out_data:
|
||||
kfree(kernel_dev);
|
||||
out_dev:
|
||||
kfree(kernel_type);
|
||||
out_type:
|
||||
return ret;
|
||||
}
|
||||
|
||||
__SYSCALL_DEFINEx(2, _qtfs_umount, char __user *, name, int, flags)
|
||||
{
|
||||
// basic validate checks done first
|
||||
if (flags & ~(MNT_FORCE | MNT_DETACH | MNT_EXPIRE | UMOUNT_NOFOLLOW))
|
||||
return -EINVAL;
|
||||
|
||||
/* if umount path is qtfs and not qtfs root, then do remote umount */
|
||||
if (qtfs_fstype_judgment(name) && !qtfs_root_judgment(name)) {
|
||||
return qtfs_remote_umount(name, flags);
|
||||
}
|
||||
|
||||
return qtfs_syscall_umount(name, flags);
|
||||
}
|
||||
|
||||
int qtfs_dir_to_qtdir(char *dir, char *qtdir, size_t len)
|
||||
{
|
||||
int ret = 0;
|
||||
struct path path;
|
||||
|
||||
if (strlen(dir) + 1 > len) {
|
||||
strlcpy(qtdir, dir, len);
|
||||
return -EINVAL;
|
||||
}
|
||||
ret = kern_path(dir, 0, &path);
|
||||
if (ret) {
|
||||
strlcpy(qtdir, dir, len);
|
||||
return 0;
|
||||
}
|
||||
if (strcmp(path.mnt->mnt_sb->s_type->name, QTFS_FSTYPE_NAME)) {
|
||||
strlcpy(qtdir, dir, len);
|
||||
} else {
|
||||
ret = qtfs_fullname(qtdir, path.dentry, len);
|
||||
}
|
||||
path_put(&path);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static size_t qtfs_strlen(const char *s)
|
||||
{
|
||||
if (s == NULL)
|
||||
return 0;
|
||||
return strlen(s);
|
||||
}
|
||||
|
||||
static long qtfs_remote_mount(char *dev_name, char __user *dir_name, char *type,
|
||||
unsigned long flags, void *data)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = qtfs_conn_get_param();
|
||||
struct qtreq_sysmount *req;
|
||||
struct qtrsp_sysmount *rsp = NULL;
|
||||
char *kernel_dir;
|
||||
int ret;
|
||||
size_t totallen;
|
||||
|
||||
if (!pvar) {
|
||||
qtfs_err("Failed to get qtfs sock var\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
kernel_dir = qtfs_copy_mount_string(dir_name);
|
||||
if (IS_ERR_OR_NULL(kernel_dir)) {
|
||||
qtfs_conn_put_param(pvar);
|
||||
return -EINVAL;
|
||||
}
|
||||
totallen = qtfs_strlen(dev_name) + qtfs_strlen(kernel_dir) + qtfs_strlen(type) + qtfs_strlen(data) + 4;
|
||||
if (totallen > sizeof(req->buf)) {
|
||||
qtfs_err("qtfs remote mount devname:%s, dir_name:%s failed, options too long.\n", dev_name, kernel_dir);
|
||||
kfree(kernel_dir);
|
||||
qtfs_conn_put_param(pvar);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
if (dev_name != NULL) {
|
||||
qtfs_dir_to_qtdir(dev_name, req->buf, sizeof(req->buf));
|
||||
req->d.dev_len = strlen(dev_name) + 1;
|
||||
} else {
|
||||
req->d.dev_len = 0;
|
||||
}
|
||||
|
||||
qtfs_dir_to_qtdir(kernel_dir, &req->buf[req->d.dev_len], sizeof(req->buf) - req->d.dev_len);
|
||||
req->d.dir_len = strlen(&req->buf[req->d.dev_len]) + 1;
|
||||
if (type != NULL) {
|
||||
strlcpy(&req->buf[req->d.dev_len + req->d.dir_len], type, strlen(type) + 1);
|
||||
req->d.type_len = strlen(type) + 1;
|
||||
} else {
|
||||
req->d.type_len = 0;
|
||||
}
|
||||
|
||||
if (data != NULL) {
|
||||
req->d.data_len = strlen(data) + 1;
|
||||
strlcpy(&req->buf[req->d.dev_len + req->d.dir_len + req->d.type_len], data, strlen(data) + 1);
|
||||
} else {
|
||||
req->d.data_len = 0;
|
||||
}
|
||||
req->d.flags = flags;
|
||||
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_SYSMOUNT, sizeof(struct qtreq_sysmount) - sizeof(req->buf) + totallen);
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
kfree(kernel_dir);
|
||||
qtfs_conn_put_param(pvar);
|
||||
return QTFS_PTR_ERR(rsp);
|
||||
}
|
||||
if (rsp->errno < 0) {
|
||||
qtfs_err("qtfs remote mount failed, devname:%s dir_name:%s type:%s, data:%s, flags(0x%lx), errno:%d\n",
|
||||
dev_name, kernel_dir, type, (char *)data, flags, rsp->errno);
|
||||
} else {
|
||||
qtfs_info("qtfs remote mount success devname:%s dir_name:%s type:%s, data:%s, flags(0x%lx)\n",
|
||||
dev_name, kernel_dir, type, (char *)data, flags);
|
||||
}
|
||||
|
||||
kfree(kernel_dir);
|
||||
ret = rsp->errno;
|
||||
qtfs_conn_put_param(pvar);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int qtfs_remote_umount(char __user *name, int flags)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = qtfs_conn_get_param();
|
||||
struct qtreq_sysumount *req;
|
||||
struct qtrsp_sysumount *rsp;
|
||||
char *kernel_name;
|
||||
int ret;
|
||||
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("qtfs remote umount get pvar failed.");
|
||||
return -EINVAL;
|
||||
}
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
kernel_name = qtfs_copy_mount_string(name);
|
||||
if (IS_ERR_OR_NULL(kernel_name)) {
|
||||
qtfs_conn_put_param(pvar);
|
||||
return QTFS_PTR_ERR(kernel_name);
|
||||
}
|
||||
req->flags = flags;
|
||||
qtfs_dir_to_qtdir(kernel_name, req->buf, sizeof(req->buf));
|
||||
qtfs_info("qtfs remote umount string:%s reqbuf:%s", (kernel_name == NULL) ? "INVALID":kernel_name, req->buf);
|
||||
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_SYSUMOUNT, sizeof(struct qtreq_sysumount) - sizeof(req->buf) + strlen(req->buf));
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
kfree(kernel_name);
|
||||
qtfs_conn_put_param(pvar);
|
||||
return QTFS_PTR_ERR(rsp);
|
||||
}
|
||||
if (rsp->errno)
|
||||
qtfs_err("qtfs remote umount failed, errno:%d\n", rsp->errno);
|
||||
|
||||
kfree(kernel_name);
|
||||
ret = rsp->errno;
|
||||
qtfs_conn_put_param(pvar);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static atomic_t replace_available = ATOMIC_INIT(1);
|
||||
|
||||
int qtfs_syscall_init(void)
|
||||
{
|
||||
if (!atomic_dec_and_test(&replace_available)) {
|
||||
atomic_inc(&replace_available);
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
symbols_origin[SYMBOL_SYSCALL_MOUNT] = qtfs_kern_syms.sys_call_table[__NR_mount];
|
||||
symbols_origin[SYMBOL_SYSCALL_UMOUNT] = qtfs_kern_syms.sys_call_table[__NR_umount2];
|
||||
symbols_origin[SYMBOL_SYSCALL_EPOLL_CTL] = qtfs_kern_syms.sys_call_table[__NR_epoll_ctl];
|
||||
#ifdef __x86_64__
|
||||
make_rw((unsigned long)qtfs_kern_syms.sys_call_table);
|
||||
qtfs_kern_syms.sys_call_table[__NR_mount] = (unsigned long *)__x64_sys_qtfs_mount;
|
||||
qtfs_kern_syms.sys_call_table[__NR_umount2] = (unsigned long *)__x64_sys_qtfs_umount;
|
||||
qtfs_kern_syms.sys_call_table[__NR_epoll_ctl] = (unsigned long *)__x64_sys_qtfs_epoll_ctl;
|
||||
make_ro((unsigned long)qtfs_kern_syms.sys_call_table);
|
||||
#endif
|
||||
#ifdef __aarch64__
|
||||
// disable write protection
|
||||
update_mapping_prot(__pa_symbol(start_rodata), (unsigned long)start_rodata, section_size, PAGE_KERNEL);
|
||||
qtfs_kern_syms.sys_call_table[__NR_mount] = (unsigned long *)__arm64_sys_qtfs_mount;
|
||||
qtfs_kern_syms.sys_call_table[__NR_umount2] = (unsigned long *)__arm64_sys_qtfs_umount;
|
||||
qtfs_kern_syms.sys_call_table[__NR_epoll_ctl] = (unsigned long *)__arm64_sys_qtfs_epoll_ctl;
|
||||
// enable write protection
|
||||
update_mapping_prot(__pa_symbol(start_rodata), (unsigned long)start_rodata, section_size, PAGE_KERNEL_RO);
|
||||
#endif
|
||||
qtfs_debug("qtfs use qtfs_mount instead of mount and umount\n");
|
||||
qtfs_debug("qtfs use qtfs_epoll_ctl instead of epoll_ctl\n");
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qtfs_syscall_fini(void)
|
||||
{
|
||||
#ifdef __x86_64__
|
||||
make_rw((unsigned long)qtfs_kern_syms.sys_call_table);
|
||||
qtfs_kern_syms.sys_call_table[__NR_mount] = (unsigned long *)symbols_origin[SYMBOL_SYSCALL_MOUNT];
|
||||
qtfs_kern_syms.sys_call_table[__NR_umount2] = (unsigned long *)symbols_origin[SYMBOL_SYSCALL_UMOUNT];
|
||||
qtfs_kern_syms.sys_call_table[__NR_epoll_ctl] = (unsigned long *)symbols_origin[SYMBOL_SYSCALL_EPOLL_CTL];
|
||||
/*set mkdir syscall to the original one */
|
||||
make_ro((unsigned long)qtfs_kern_syms.sys_call_table);
|
||||
#endif
|
||||
#ifdef __aarch64__
|
||||
// disable write protection
|
||||
update_mapping_prot(__pa_symbol(start_rodata), (unsigned long)start_rodata, section_size, PAGE_KERNEL);
|
||||
qtfs_kern_syms.sys_call_table[__NR_mount] = (unsigned long *)symbols_origin[SYMBOL_SYSCALL_MOUNT];
|
||||
qtfs_kern_syms.sys_call_table[__NR_umount2] = (unsigned long *)symbols_origin[SYMBOL_SYSCALL_UMOUNT];
|
||||
qtfs_kern_syms.sys_call_table[__NR_epoll_ctl] = (unsigned long *)symbols_origin[SYMBOL_SYSCALL_EPOLL_CTL];
|
||||
// enable write protection
|
||||
update_mapping_prot(__pa_symbol(start_rodata), (unsigned long)start_rodata, section_size, PAGE_KERNEL_RO);
|
||||
#endif
|
||||
qtfs_info("qtfs mount umount and epoll_ctl resumed\n");
|
||||
atomic_inc(&replace_available);
|
||||
return 0;
|
||||
}
|
20
qtfs/qtfs/syscall.h
Normal file
20
qtfs/qtfs/syscall.h
Normal file
@ -0,0 +1,20 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_SYSCALL_H__
|
||||
#define __QTFS_SYSCALL_H__
|
||||
|
||||
extern int qtfs_syscall_init(void);
|
||||
extern int qtfs_syscall_fini(void);
|
||||
|
||||
#endif
|
318
qtfs/qtfs/xattr.c
Normal file
318
qtfs/qtfs/xattr.c
Normal file
@ -0,0 +1,318 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/time.h>
|
||||
#include <linux/xattr.h>
|
||||
#include <linux/version.h>
|
||||
|
||||
#include "conn.h"
|
||||
#include "qtfs-mod.h"
|
||||
#include "req.h"
|
||||
#include "log.h"
|
||||
|
||||
ssize_t qtfs_xattr_list(struct dentry *dentry, char *buffer, size_t buffer_size)
|
||||
{
|
||||
struct qtreq_xattrlist *req;
|
||||
struct qtrsp_xattrlist *rsp;
|
||||
struct qtfs_conn_var_s *pvar = qtfs_conn_get_param();
|
||||
ssize_t ret;
|
||||
|
||||
if (!pvar) {
|
||||
qtfs_err("qtfs_xattr_list Failed to get qtfs sock var");
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (dentry == NULL) {
|
||||
qtfs_err("qtfs_xattr_list dentry is NULL.");
|
||||
qtfs_conn_put_param(pvar);
|
||||
return 0;
|
||||
}
|
||||
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
if (qtfs_fullname(req->path, dentry, sizeof(req->path)) < 0) {
|
||||
qtfs_err("qtfs fullname failed");
|
||||
qtfs_conn_put_param(pvar);
|
||||
return 0;
|
||||
}
|
||||
req->buffer_size = buffer_size;
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_XATTRLIST, QTFS_SEND_SIZE(struct qtreq_xattrlist, req->path));
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
qtfs_err("qtfs_xattr_list remote run failed.");
|
||||
qtfs_conn_put_param(pvar);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (rsp->d.ret == QTFS_ERR) {
|
||||
qtfs_err("qtfs_xattr_list failed with ret:%d.", rsp->d.ret);
|
||||
ret = rsp->d.size;
|
||||
qtfs_conn_put_param(pvar);
|
||||
return ret;
|
||||
}
|
||||
ret = rsp->d.size;
|
||||
if (buffer != NULL) {
|
||||
ret = (rsp->d.size > buffer_size) ? buffer_size : rsp->d.size;
|
||||
memcpy(buffer, rsp->name, ret);
|
||||
}
|
||||
qtfs_conn_put_param(pvar);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int qtfs_xattr_set(const struct xattr_handler *handler,
|
||||
struct dentry *dentry, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags);
|
||||
|
||||
static int
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(6, 4, 0))
|
||||
qtfs_xattr_user_set(const struct xattr_handler *handler,
|
||||
struct mnt_idmap *idmap,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
qtfs_xattr_user_set(const struct xattr_handler *handler,
|
||||
struct user_namespace *mnt_userns,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#else
|
||||
qtfs_xattr_user_set(const struct xattr_handler *handler,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#endif
|
||||
|
||||
{
|
||||
return qtfs_xattr_set(handler, unused, inode, name, value, size, flags);
|
||||
}
|
||||
|
||||
static int
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(6, 4, 0))
|
||||
qtfs_xattr_trusted_set(const struct xattr_handler *handler,
|
||||
struct mnt_idmap *idmap,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
qtfs_xattr_trusted_set(const struct xattr_handler *handler,
|
||||
struct user_namespace *mnt_userns,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#else
|
||||
qtfs_xattr_trusted_set(const struct xattr_handler *handler,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#endif
|
||||
{
|
||||
return qtfs_xattr_set(handler, unused, inode, name, value, size, flags);
|
||||
}
|
||||
|
||||
static int
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(6, 4, 0))
|
||||
qtfs_xattr_security_set(const struct xattr_handler *handler,
|
||||
struct mnt_idmap *idmap,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
qtfs_xattr_security_set(const struct xattr_handler *handler,
|
||||
struct user_namespace *mnt_userns,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#else
|
||||
qtfs_xattr_security_set(const struct xattr_handler *handler,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#endif
|
||||
{
|
||||
return qtfs_xattr_set(handler, unused, inode, name, value, size, flags);
|
||||
}
|
||||
|
||||
static int qtfs_xattr_set(const struct xattr_handler *handler,
|
||||
struct dentry *dentry, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
{
|
||||
struct qtreq_xattrset *req;
|
||||
struct qtrsp_xattrset *rsp;
|
||||
struct qtfs_conn_var_s *pvar = qtfs_conn_get_param();
|
||||
int ret;
|
||||
size_t totallen;
|
||||
|
||||
if (!pvar) {
|
||||
qtfs_err("failed to get qtfs sock var");
|
||||
return -ENOMEM;
|
||||
}
|
||||
if (dentry == NULL) {
|
||||
qtfs_conn_put_param(pvar);
|
||||
return -ENOENT;
|
||||
}
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
if (qtfs_fullname(req->buf, dentry, sizeof(req->buf)) < 0) {
|
||||
qtfs_err("xattr set get fullname failed.");
|
||||
qtfs_conn_put_param(pvar);
|
||||
return -EFAULT;
|
||||
}
|
||||
req->d.valuelen = size;
|
||||
req->d.flags = flags;
|
||||
req->d.pathlen = strlen(req->buf) + 1;
|
||||
req->d.namelen = strlen(name) + strlen(handler->prefix) + 1;
|
||||
qtfs_info("xattr set path:%s name:%s size:%lu", req->buf, name, size);
|
||||
totallen = req->d.pathlen + req->d.namelen + size;
|
||||
if (totallen >= sizeof(req->buf)) {
|
||||
qtfs_err("xattr set namelen:%lu size:%lu is too long", req->d.namelen, size);
|
||||
qtfs_conn_put_param(pvar);
|
||||
return -EFAULT;
|
||||
}
|
||||
strlcpy(&req->buf[req->d.pathlen], handler->prefix, strlen(handler->prefix) + 1);
|
||||
strcat(&req->buf[req->d.pathlen], name);
|
||||
if (size > 0) {
|
||||
memcpy(&req->buf[req->d.pathlen + req->d.namelen], value, size);
|
||||
req->d.valuelen = size + 1;
|
||||
}
|
||||
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_XATTRSET, sizeof(struct qtreq_xattrset) - sizeof(req->buf) + req->d.pathlen + req->d.namelen + req->d.valuelen);
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
qtfs_conn_put_param(pvar);
|
||||
return QTFS_PTR_ERR(rsp);
|
||||
}
|
||||
if (rsp->errno < 0) {
|
||||
qtfs_err("xattr set failed file:%s name:%s", req->buf, name);
|
||||
} else {
|
||||
qtfs_info("xattr set successed file:%s name:%s", req->buf, name);
|
||||
}
|
||||
ret = rsp->errno;
|
||||
qtfs_conn_put_param(pvar);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int qtfs_xattr_get(const struct xattr_handler *handler,
|
||||
struct dentry *dentry, struct inode *inode,
|
||||
const char *name, void *buffer, size_t size)
|
||||
{
|
||||
struct qtreq_xattrget *req;
|
||||
struct qtrsp_xattrget *rsp;
|
||||
struct qtfs_conn_var_s *pvar = qtfs_conn_get_param();
|
||||
size_t leftlen = size;
|
||||
char *buf = (char *)buffer;
|
||||
|
||||
if (!pvar) {
|
||||
qtfs_err("Failed to get qtfs sock var");
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (dentry == NULL) {
|
||||
qtfs_err("xattr get dentry is NULL.");
|
||||
qtfs_conn_put_param(pvar);
|
||||
return 0;
|
||||
}
|
||||
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
if (qtfs_fullname(req->path, dentry, sizeof(req->path)) < 0) {
|
||||
qtfs_err("qtfs fullname failed");
|
||||
qtfs_conn_put_param(pvar);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (strlen(handler->prefix) + strlen(name) <= (sizeof(req->d.prefix_name) - 1)) {
|
||||
strcpy(req->d.prefix_name, handler->prefix);
|
||||
strcat(req->d.prefix_name, name);
|
||||
} else {
|
||||
qtfs_err("strcpy len too long");
|
||||
qtfs_conn_put_param(pvar);
|
||||
return 0;
|
||||
}
|
||||
|
||||
rsp = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_RECV);
|
||||
do {
|
||||
req->d.pos = rsp->d.pos;
|
||||
req->d.size = size;
|
||||
rsp = qtfs_remote_run(pvar, QTFS_REQ_XATTRGET, QTFS_SEND_SIZE(struct qtreq_xattrget, req->path));
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
qtfs_err("rsp invalid, file:%s", req->path);
|
||||
qtfs_conn_put_param(pvar);
|
||||
return QTFS_PTR_ERR(rsp);
|
||||
}
|
||||
if (rsp->d.ret == QTFS_ERR || (size !=0 && (rsp->d.size > req->d.size || leftlen < rsp->d.size))) {
|
||||
qtfs_err("ret:%d rsp size:%ld req size:%d leftlen:%lu", rsp->d.ret, rsp->d.size,
|
||||
req->d.size, leftlen);
|
||||
goto err_end;
|
||||
}
|
||||
if (size > 0 && rsp->d.size <= leftlen) {
|
||||
memcpy(&buf[size - leftlen], rsp->buf, rsp->d.size);
|
||||
}
|
||||
leftlen -= rsp->d.size;
|
||||
} while (leftlen > 0 && rsp->d.size > 0);
|
||||
qtfs_info("qtfs getxattr success:<<%s>>", buf);
|
||||
|
||||
qtfs_conn_put_param(pvar);
|
||||
|
||||
return size - leftlen;
|
||||
|
||||
err_end:
|
||||
qtfs_conn_put_param(pvar);
|
||||
return -ENODATA;
|
||||
}
|
||||
|
||||
const struct xattr_handler qtfs_xattr_user_handler = {
|
||||
.prefix = XATTR_USER_PREFIX,
|
||||
.get = qtfs_xattr_get,
|
||||
.set = qtfs_xattr_user_set,
|
||||
};
|
||||
|
||||
const struct xattr_handler qtfs_xattr_trusted_handler = {
|
||||
.prefix = XATTR_TRUSTED_PREFIX,
|
||||
.get = qtfs_xattr_get,
|
||||
.set = qtfs_xattr_trusted_set,
|
||||
};
|
||||
|
||||
const struct xattr_handler qtfs_xattr_security_handler = {
|
||||
.prefix = XATTR_SECURITY_PREFIX,
|
||||
.get = qtfs_xattr_get,
|
||||
.set = qtfs_xattr_security_set,
|
||||
};
|
||||
|
||||
#ifndef KVER_4_19
|
||||
static int
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(6, 4, 0))
|
||||
qtfs_xattr_hurd_set(const struct xattr_handler *handler,
|
||||
struct mnt_idmap *idmap,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0))
|
||||
qtfs_xattr_hurd_set(const struct xattr_handler *handler,
|
||||
struct user_namespace *mnt_userns,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#else
|
||||
qtfs_xattr_hurd_set(const struct xattr_handler *handler,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
const char *name, const void *value,
|
||||
size_t size, int flags)
|
||||
#endif
|
||||
{
|
||||
return qtfs_xattr_set(handler, unused, inode, name, value, size, flags);
|
||||
}
|
||||
|
||||
const struct xattr_handler qtfs_xattr_hurd_handler = {
|
||||
.prefix = XATTR_HURD_PREFIX,
|
||||
.get = qtfs_xattr_get,
|
||||
.set = qtfs_xattr_hurd_set,
|
||||
};
|
||||
#endif
|
340
qtfs/qtfs_common/License
Normal file
340
qtfs/qtfs_common/License
Normal file
@ -0,0 +1,340 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 2, June 1991
|
||||
|
||||
Copyright (C) 1989, 1991 Free Software Foundation, Inc.
|
||||
51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The licenses for most software are designed to take away your
|
||||
freedom to share and change it. By contrast, the GNU General Public
|
||||
License is intended to guarantee your freedom to share and change free
|
||||
software--to make sure the software is free for all its users. This
|
||||
General Public License applies to most of the Free Software
|
||||
Foundation's software and to any other program whose authors commit to
|
||||
using it. (Some other Free Software Foundation software is covered by
|
||||
the GNU Library General Public License instead.) You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
this service if you wish), that you receive source code or can get it
|
||||
if you want it, that you can change the software or use pieces of it
|
||||
in new free programs; and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to make restrictions that forbid
|
||||
anyone to deny you these rights or to ask you to surrender the rights.
|
||||
These restrictions translate to certain responsibilities for you if you
|
||||
distribute copies of the software, or if you modify it.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must give the recipients all the rights that
|
||||
you have. You must make sure that they, too, receive or can get the
|
||||
source code. And you must show them these terms so they know their
|
||||
rights.
|
||||
|
||||
We protect your rights with two steps: (1) copyright the software, and
|
||||
(2) offer you this license which gives you legal permission to copy,
|
||||
distribute and/or modify the software.
|
||||
|
||||
Also, for each author's protection and ours, we want to make certain
|
||||
that everyone understands that there is no warranty for this free
|
||||
software. If the software is modified by someone else and passed on, we
|
||||
want its recipients to know that what they have is not the original, so
|
||||
that any problems introduced by others will not reflect on the original
|
||||
authors' reputations.
|
||||
|
||||
Finally, any free program is threatened constantly by software
|
||||
patents. We wish to avoid the danger that redistributors of a free
|
||||
program will individually obtain patent licenses, in effect making the
|
||||
program proprietary. To prevent this, we have made it clear that any
|
||||
patent must be licensed for everyone's free use or not licensed at all.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||
|
||||
0. This License applies to any program or other work which contains
|
||||
a notice placed by the copyright holder saying it may be distributed
|
||||
under the terms of this General Public License. The "Program", below,
|
||||
refers to any such program or work, and a "work based on the Program"
|
||||
means either the Program or any derivative work under copyright law:
|
||||
that is to say, a work containing the Program or a portion of it,
|
||||
either verbatim or with modifications and/or translated into another
|
||||
language. (Hereinafter, translation is included without limitation in
|
||||
the term "modification".) Each licensee is addressed as "you".
|
||||
|
||||
Activities other than copying, distribution and modification are not
|
||||
covered by this License; they are outside its scope. The act of
|
||||
running the Program is not restricted, and the output from the Program
|
||||
is covered only if its contents constitute a work based on the
|
||||
Program (independent of having been made by running the Program).
|
||||
Whether that is true depends on what the Program does.
|
||||
|
||||
1. You may copy and distribute verbatim copies of the Program's
|
||||
source code as you receive it, in any medium, provided that you
|
||||
conspicuously and appropriately publish on each copy an appropriate
|
||||
copyright notice and disclaimer of warranty; keep intact all the
|
||||
notices that refer to this License and to the absence of any warranty;
|
||||
and give any other recipients of the Program a copy of this License
|
||||
along with the Program.
|
||||
|
||||
You may charge a fee for the physical act of transferring a copy, and
|
||||
you may at your option offer warranty protection in exchange for a fee.
|
||||
|
||||
2. You may modify your copy or copies of the Program or any portion
|
||||
of it, thus forming a work based on the Program, and copy and
|
||||
distribute such modifications or work under the terms of Section 1
|
||||
above, provided that you also meet all of these conditions:
|
||||
|
||||
a) You must cause the modified files to carry prominent notices
|
||||
stating that you changed the files and the date of any change.
|
||||
|
||||
b) You must cause any work that you distribute or publish, that in
|
||||
whole or in part contains or is derived from the Program or any
|
||||
part thereof, to be licensed as a whole at no charge to all third
|
||||
parties under the terms of this License.
|
||||
|
||||
c) If the modified program normally reads commands interactively
|
||||
when run, you must cause it, when started running for such
|
||||
interactive use in the most ordinary way, to print or display an
|
||||
announcement including an appropriate copyright notice and a
|
||||
notice that there is no warranty (or else, saying that you provide
|
||||
a warranty) and that users may redistribute the program under
|
||||
these conditions, and telling the user how to view a copy of this
|
||||
License. (Exception: if the Program itself is interactive but
|
||||
does not normally print such an announcement, your work based on
|
||||
the Program is not required to print an announcement.)
|
||||
|
||||
These requirements apply to the modified work as a whole. If
|
||||
identifiable sections of that work are not derived from the Program,
|
||||
and can be reasonably considered independent and separate works in
|
||||
themselves, then this License, and its terms, do not apply to those
|
||||
sections when you distribute them as separate works. But when you
|
||||
distribute the same sections as part of a whole which is a work based
|
||||
on the Program, the distribution of the whole must be on the terms of
|
||||
this License, whose permissions for other licensees extend to the
|
||||
entire whole, and thus to each and every part regardless of who wrote it.
|
||||
|
||||
Thus, it is not the intent of this section to claim rights or contest
|
||||
your rights to work written entirely by you; rather, the intent is to
|
||||
exercise the right to control the distribution of derivative or
|
||||
collective works based on the Program.
|
||||
|
||||
In addition, mere aggregation of another work not based on the Program
|
||||
with the Program (or with a work based on the Program) on a volume of
|
||||
a storage or distribution medium does not bring the other work under
|
||||
the scope of this License.
|
||||
|
||||
3. You may copy and distribute the Program (or a work based on it,
|
||||
under Section 2) in object code or executable form under the terms of
|
||||
Sections 1 and 2 above provided that you also do one of the following:
|
||||
|
||||
a) Accompany it with the complete corresponding machine-readable
|
||||
source code, which must be distributed under the terms of Sections
|
||||
1 and 2 above on a medium customarily used for software interchange; or,
|
||||
|
||||
b) Accompany it with a written offer, valid for at least three
|
||||
years, to give any third party, for a charge no more than your
|
||||
cost of physically performing source distribution, a complete
|
||||
machine-readable copy of the corresponding source code, to be
|
||||
distributed under the terms of Sections 1 and 2 above on a medium
|
||||
customarily used for software interchange; or,
|
||||
|
||||
c) Accompany it with the information you received as to the offer
|
||||
to distribute corresponding source code. (This alternative is
|
||||
allowed only for noncommercial distribution and only if you
|
||||
received the program in object code or executable form with such
|
||||
an offer, in accord with Subsection b above.)
|
||||
|
||||
The source code for a work means the preferred form of the work for
|
||||
making modifications to it. For an executable work, complete source
|
||||
code means all the source code for all modules it contains, plus any
|
||||
associated interface definition files, plus the scripts used to
|
||||
control compilation and installation of the executable. However, as a
|
||||
special exception, the source code distributed need not include
|
||||
anything that is normally distributed (in either source or binary
|
||||
form) with the major components (compiler, kernel, and so on) of the
|
||||
operating system on which the executable runs, unless that component
|
||||
itself accompanies the executable.
|
||||
|
||||
If distribution of executable or object code is made by offering
|
||||
access to copy from a designated place, then offering equivalent
|
||||
access to copy the source code from the same place counts as
|
||||
distribution of the source code, even though third parties are not
|
||||
compelled to copy the source along with the object code.
|
||||
|
||||
4. You may not copy, modify, sublicense, or distribute the Program
|
||||
except as expressly provided under this License. Any attempt
|
||||
otherwise to copy, modify, sublicense or distribute the Program is
|
||||
void, and will automatically terminate your rights under this License.
|
||||
However, parties who have received copies, or rights, from you under
|
||||
this License will not have their licenses terminated so long as such
|
||||
parties remain in full compliance.
|
||||
|
||||
5. You are not required to accept this License, since you have not
|
||||
signed it. However, nothing else grants you permission to modify or
|
||||
distribute the Program or its derivative works. These actions are
|
||||
prohibited by law if you do not accept this License. Therefore, by
|
||||
modifying or distributing the Program (or any work based on the
|
||||
Program), you indicate your acceptance of this License to do so, and
|
||||
all its terms and conditions for copying, distributing or modifying
|
||||
the Program or works based on it.
|
||||
|
||||
6. Each time you redistribute the Program (or any work based on the
|
||||
Program), the recipient automatically receives a license from the
|
||||
original licensor to copy, distribute or modify the Program subject to
|
||||
these terms and conditions. You may not impose any further
|
||||
restrictions on the recipients' exercise of the rights granted herein.
|
||||
You are not responsible for enforcing compliance by third parties to
|
||||
this License.
|
||||
|
||||
7. If, as a consequence of a court judgment or allegation of patent
|
||||
infringement or for any other reason (not limited to patent issues),
|
||||
conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot
|
||||
distribute so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you
|
||||
may not distribute the Program at all. For example, if a patent
|
||||
license would not permit royalty-free redistribution of the Program by
|
||||
all those who receive copies directly or indirectly through you, then
|
||||
the only way you could satisfy both it and this License would be to
|
||||
refrain entirely from distribution of the Program.
|
||||
|
||||
If any portion of this section is held invalid or unenforceable under
|
||||
any particular circumstance, the balance of the section is intended to
|
||||
apply and the section as a whole is intended to apply in other
|
||||
circumstances.
|
||||
|
||||
It is not the purpose of this section to induce you to infringe any
|
||||
patents or other property right claims or to contest validity of any
|
||||
such claims; this section has the sole purpose of protecting the
|
||||
integrity of the free software distribution system, which is
|
||||
implemented by public license practices. Many people have made
|
||||
generous contributions to the wide range of software distributed
|
||||
through that system in reliance on consistent application of that
|
||||
system; it is up to the author/donor to decide if he or she is willing
|
||||
to distribute software through any other system and a licensee cannot
|
||||
impose that choice.
|
||||
|
||||
This section is intended to make thoroughly clear what is believed to
|
||||
be a consequence of the rest of this License.
|
||||
|
||||
8. If the distribution and/or use of the Program is restricted in
|
||||
certain countries either by patents or by copyrighted interfaces, the
|
||||
original copyright holder who places the Program under this License
|
||||
may add an explicit geographical distribution limitation excluding
|
||||
those countries, so that distribution is permitted only in or among
|
||||
countries not thus excluded. In such case, this License incorporates
|
||||
the limitation as if written in the body of this License.
|
||||
|
||||
9. The Free Software Foundation may publish revised and/or new versions
|
||||
of the General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the Program
|
||||
specifies a version number of this License which applies to it and "any
|
||||
later version", you have the option of following the terms and conditions
|
||||
either of that version or of any later version published by the Free
|
||||
Software Foundation. If the Program does not specify a version number of
|
||||
this License, you may choose any version ever published by the Free Software
|
||||
Foundation.
|
||||
|
||||
10. If you wish to incorporate parts of the Program into other free
|
||||
programs whose distribution conditions are different, write to the author
|
||||
to ask for permission. For software which is copyrighted by the Free
|
||||
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||
make exceptions for this. Our decision will be guided by the two goals
|
||||
of preserving the free status of all derivatives of our free software and
|
||||
of promoting the sharing and reuse of software generally.
|
||||
|
||||
NO WARRANTY
|
||||
|
||||
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||
REPAIR OR CORRECTION.
|
||||
|
||||
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||
POSSIBILITY OF SUCH DAMAGES.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
convey the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation; either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program; if not, write to the Free Software
|
||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program is interactive, make it output a short notice like this
|
||||
when it starts in an interactive mode:
|
||||
|
||||
Gnomovision version 69, Copyright (C) year name of author
|
||||
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, the commands you use may
|
||||
be called something other than `show w' and `show c'; they could even be
|
||||
mouse-clicks or menu items--whatever suits your program.
|
||||
|
||||
You should also get your employer (if you work as a programmer) or your
|
||||
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||
necessary. Here is a sample; alter the names:
|
||||
|
||||
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||
|
||||
<signature of Ty Coon>, 1 April 1989
|
||||
Ty Coon, President of Vice
|
||||
|
||||
This General Public License does not permit incorporating your program into
|
||||
proprietary programs. If your program is a subroutine library, you may
|
||||
consider it more useful to permit linking proprietary applications with the
|
||||
library. If this is what you want to do, use the GNU Library General
|
||||
Public License instead of this License.
|
951
qtfs/qtfs_common/conn.c
Normal file
951
qtfs/qtfs_common/conn.c
Normal file
@ -0,0 +1,951 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/kallsyms.h>
|
||||
#include <linux/tcp.h>
|
||||
#include <net/tcp.h>
|
||||
#include <linux/un.h>
|
||||
#include <linux/file.h>
|
||||
|
||||
#include "comm.h"
|
||||
#include "conn.h"
|
||||
#include "log.h"
|
||||
#include "req.h"
|
||||
#include "symbol_wrapper.h"
|
||||
#include "uds_module.h"
|
||||
|
||||
struct qtfs_pvar_ops_s *g_pvar_ops = NULL;
|
||||
char qtfs_log_level[QTFS_LOGLEVEL_STRLEN] = {0};
|
||||
char qtfs_conn_type[20] = QTFS_CONN_SOCK_TYPE;
|
||||
int log_level = LOG_ERROR;
|
||||
static int qtfs_conn_max_conn = QTFS_MAX_THREADS;
|
||||
struct qtinfo *qtfs_diag_info = NULL;
|
||||
bool qtfs_epoll_mode = false; // true: support any mode; false: only support fifo
|
||||
|
||||
static atomic_t g_qtfs_conn_num;
|
||||
static struct list_head g_vld_lst;
|
||||
static struct list_head g_busy_lst;
|
||||
static struct llist_head g_lazy_put_llst;
|
||||
static struct list_head g_fifo_lst;
|
||||
static struct mutex g_param_mutex;
|
||||
static struct mutex g_fifo_mutex;
|
||||
int qtfs_mod_exiting = false;
|
||||
struct qtfs_conn_var_s *qtfs_thread_var[QTFS_MAX_THREADS] = {NULL};
|
||||
struct qtfs_conn_var_s *qtfs_epoll_var = NULL;
|
||||
#ifdef QTFS_SERVER
|
||||
struct qtfs_server_userp_s *qtfs_userps = NULL;
|
||||
#endif
|
||||
#ifdef QTFS_CLIENT
|
||||
struct kmem_cache *qtfs_fifo_pvar_cache;
|
||||
#endif
|
||||
|
||||
// try to connect remote uds server, only for unix domain socket
|
||||
#define QTFS_UDS_PROXY_SUFFIX ".proxy"
|
||||
int qtfs_uds_proxy_build(struct socket *sock, struct sockaddr_un *addr, int len)
|
||||
{
|
||||
int ret;
|
||||
struct uds_proxy_remote_conn_req req;
|
||||
struct uds_proxy_remote_conn_rsp rsp;
|
||||
struct sockaddr_un proxy = {.sun_family = AF_UNIX};
|
||||
struct socket *proxy_sock;
|
||||
struct msghdr msgs;
|
||||
struct msghdr msgr;
|
||||
struct kvec vec;
|
||||
|
||||
ret = sock_create_kern(&init_net, AF_UNIX, SOCK_STREAM, 0, &proxy_sock);
|
||||
if (ret) {
|
||||
qtfs_err("create proxy sock failed sun path:%s", addr->sun_path);
|
||||
return -EFAULT;
|
||||
}
|
||||
memset(proxy.sun_path, 0, sizeof(proxy.sun_path));
|
||||
strlcpy(proxy.sun_path, UDS_BUILD_CONN_ADDR, strlen(UDS_BUILD_CONN_ADDR) + 1);
|
||||
ret = sock->ops->connect(proxy_sock, (struct sockaddr *)&proxy, sizeof(proxy), SOCK_NONBLOCK);
|
||||
if (ret) {
|
||||
qtfs_err("connect to uds proxy failed");
|
||||
goto err_end;
|
||||
}
|
||||
memset(req.sun_path, 0, sizeof(req.sun_path));
|
||||
strlcpy(req.sun_path, addr->sun_path, sizeof(req.sun_path));
|
||||
memset(&msgs, 0, sizeof(struct msghdr));
|
||||
memset(&msgr, 0, sizeof(struct msghdr));
|
||||
req.type = sock->sk->sk_type;
|
||||
vec.iov_base = &req;
|
||||
vec.iov_len = sizeof(req);
|
||||
ret = kernel_sendmsg(proxy_sock, &msgs, &vec, 1, vec.iov_len);
|
||||
if (ret < 0) {
|
||||
qtfs_err("send remote connect request failed:%d", ret);
|
||||
goto err_end;
|
||||
}
|
||||
vec.iov_base = &rsp;
|
||||
vec.iov_len = sizeof(rsp);
|
||||
ret = kernel_recvmsg(proxy_sock, &msgr, &vec, 1, vec.iov_len, MSG_WAITALL);
|
||||
if (ret <= 0) {
|
||||
qtfs_err("recv remote connect response failed:%d", ret);
|
||||
goto err_end;
|
||||
}
|
||||
if (rsp.ret == 0) {
|
||||
goto err_end;
|
||||
}
|
||||
qtfs_info("try to build uds proxy successed, sun path:%s", addr->sun_path);
|
||||
|
||||
sock_release(proxy_sock);
|
||||
return 0;
|
||||
err_end:
|
||||
sock_release(proxy_sock);
|
||||
return -ECONNREFUSED;
|
||||
}
|
||||
|
||||
static int qtfs_uds_remote_whitelist(const char *path)
|
||||
{
|
||||
int i;
|
||||
int ret = 1;
|
||||
struct qtfs_wl_cap *cap;
|
||||
read_lock(&g_qtfs_wl.rwlock);
|
||||
cap = &g_qtfs_wl.cap[QTFS_WHITELIST_UDSCONNECT];
|
||||
for (i = 0; i < cap->nums; i++) {
|
||||
if (strncmp(path, cap->item[i], strlen(cap->item[i])) == 0) {
|
||||
if (strlen(path) > strlen(cap->item[i]) && path[strlen(cap->item[i])] != '/') {
|
||||
continue;
|
||||
}
|
||||
ret = 0;
|
||||
break;
|
||||
}
|
||||
}
|
||||
read_unlock(&g_qtfs_wl.rwlock);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static inline int qtfs_uds_is_proxy(void)
|
||||
{
|
||||
#define UDS_PROXYD_PRNAME "udsproxyd"
|
||||
if (strlen(current->comm) == strlen(UDS_PROXYD_PRNAME) &&
|
||||
strncmp(current->comm, UDS_PROXYD_PRNAME, strlen(UDS_PROXYD_PRNAME)) == 0)
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline int qtfs_uds_is_rexec(void)
|
||||
{
|
||||
#define REXEC_PRNAME "rexec"
|
||||
if (strlen(current->comm) == strlen(REXEC_PRNAME) &&
|
||||
strncmp(current->comm, REXEC_PRNAME, strlen(REXEC_PRNAME)) == 0)
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qtfs_uds_remote_connect_user(int fd, struct sockaddr __user *addr, int len)
|
||||
{
|
||||
int sysret = -EINVAL;
|
||||
int ret;
|
||||
int err;
|
||||
int un_headlen;
|
||||
struct fd f;
|
||||
struct socket *sock;
|
||||
struct sockaddr_un addr_un;
|
||||
struct sockaddr_un addr_proxy;
|
||||
|
||||
if (qtfs_uds_is_rexec()) {
|
||||
qtfs_info("Rexec process has no nessary to connect local server");
|
||||
goto try_conn_remote;
|
||||
}
|
||||
sysret = qtfs_syscall_connect(fd, addr, len);
|
||||
// don't try remote uds connect if: 1.local connect successed; 2.this process is udsproxyd
|
||||
if (sysret == 0 || qtfs_uds_is_proxy())
|
||||
return sysret;
|
||||
try_conn_remote:
|
||||
// len is passed from syscall input args directly. it's trustworthy
|
||||
if (copy_from_user(&addr_un, addr, len)) {
|
||||
qtfs_err("copy sockaddr failed.");
|
||||
return sysret;
|
||||
}
|
||||
// don't try remote uds connect if sunpath not in whitelist
|
||||
if (qtfs_uds_remote_whitelist(addr_un.sun_path) != 0)
|
||||
return sysret;
|
||||
if (addr_un.sun_family != AF_UNIX)
|
||||
return sysret;
|
||||
un_headlen = sizeof(struct sockaddr_un) - sizeof(addr_un.sun_path);
|
||||
// 如果用户态给的参数长度不够,这里智能失败退出
|
||||
if (len < un_headlen || strlen(addr_un.sun_path) >= (len - un_headlen - strlen(QTFS_UDS_PROXY_SUFFIX))) {
|
||||
qtfs_err("failed to try connect remote uds server, sun path:%s too long to add suffix:%s",
|
||||
addr_un.sun_path, QTFS_UDS_PROXY_SUFFIX);
|
||||
return sysret;
|
||||
}
|
||||
qtfs_info("uds connect failed:%d try to remote connect:%s.", sysret, addr_un.sun_path);
|
||||
|
||||
f = fdget(fd);
|
||||
if (f.file == NULL) {
|
||||
return -EBADF;
|
||||
}
|
||||
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 11, 0))
|
||||
sock =sock_from_file(f.file);
|
||||
#else
|
||||
sock = sock_from_file(f.file, &err);
|
||||
#endif
|
||||
if (!sock) {
|
||||
goto end;
|
||||
}
|
||||
// try to connect remote uds's proxy
|
||||
ret = qtfs_uds_proxy_build(sock, &addr_un, len);
|
||||
if (ret == 0) {
|
||||
memcpy(&addr_proxy, &addr_un, sizeof(struct sockaddr_un));
|
||||
strlcat(addr_proxy.sun_path, QTFS_UDS_PROXY_SUFFIX, sizeof(addr_proxy.sun_path));
|
||||
if (copy_to_user(addr, &addr_proxy, (len > sizeof(struct sockaddr_un)) ? sizeof(struct sockaddr_un) : len)) {
|
||||
qtfs_err("copy to addr failed sunpath:%s", addr_proxy.sun_path);
|
||||
goto end;
|
||||
}
|
||||
sysret = qtfs_syscall_connect(fd, addr, len);
|
||||
qtfs_info("try remote connect sunpath:%s ret:%d", addr_un.sun_path, sysret);
|
||||
if (copy_to_user(addr, &addr_un, (len > sizeof(struct sockaddr_un)) ? sizeof(struct sockaddr_un) : len)) {
|
||||
qtfs_err("resume addr failed");
|
||||
goto end;
|
||||
}
|
||||
}
|
||||
|
||||
end:
|
||||
fdput(f);
|
||||
return sysret;
|
||||
}
|
||||
|
||||
int qtfs_conn_init(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
return pvar->conn_ops->conn_init(&pvar->conn_var, pvar->user_type);
|
||||
}
|
||||
|
||||
void qtfs_conn_fini(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
return pvar->conn_ops->conn_fini(&pvar->conn_var, pvar->user_type);
|
||||
}
|
||||
|
||||
#define MAGIC_U32(magic, n) ((magic >> (n * 8)) & 0xff)
|
||||
static inline int qtfs_conn_sync_magic(struct qtfs_conn_var_s *pvar, bool block)
|
||||
{
|
||||
u8 byte;
|
||||
int ret;
|
||||
if (pvar->magic_recv == 0)
|
||||
return 0;
|
||||
while (1) {
|
||||
ret = pvar->conn_ops->conn_recv(&pvar->conn_var, &byte, 1, block);
|
||||
if (ret <= 0) break;
|
||||
if (byte != MAGIC_U32(pvar->magic_recv, 3)) continue;
|
||||
ret = pvar->conn_ops->conn_recv(&pvar->conn_var, &byte, 1, block);
|
||||
if (ret <= 0) break;
|
||||
if (byte != MAGIC_U32(pvar->magic_recv, 2)) continue;
|
||||
ret = pvar->conn_ops->conn_recv(&pvar->conn_var, &byte, 1, block);
|
||||
if (ret <= 0) break;
|
||||
if (byte != MAGIC_U32(pvar->magic_recv, 1)) continue;
|
||||
ret = pvar->conn_ops->conn_recv(&pvar->conn_var, &byte, 1, block);
|
||||
if (ret <= 0) break;
|
||||
if (byte != MAGIC_U32(pvar->magic_recv, 0)) continue;
|
||||
break;
|
||||
}
|
||||
if (ret < 0) {
|
||||
if (ret != -EAGAIN)
|
||||
qtfs_err("qtfs sync magic failed ret:%d byte:%u", ret, byte);
|
||||
return ret;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qtfs_conn_send(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret = 0;
|
||||
int iov_ret = 0;
|
||||
if (pvar->vec_send.iov_len > pvar->send_max)
|
||||
return -EMSGSIZE;
|
||||
if (pvar->magic_send != 0) {
|
||||
ret = pvar->conn_ops->conn_send(&pvar->conn_var, &pvar->magic_send, sizeof(pvar->magic_send));
|
||||
if (ret <= 0) {
|
||||
qtfs_err("magic send failed, ret:%d", ret);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
pvar->send_valid = pvar->vec_send.iov_len;
|
||||
ret = pvar->conn_ops->conn_send(&pvar->conn_var, pvar->vec_send.iov_base, pvar->vec_send.iov_len);
|
||||
if (ret <= 0)
|
||||
return ret;
|
||||
if (pvar->iov_send) {
|
||||
iov_ret = pvar->conn_ops->conn_send_iter(&pvar->conn_var, pvar->iov_send);
|
||||
pvar->iov_send = NULL; // invalid it after use
|
||||
if (iov_ret <= 0)
|
||||
return iov_ret;
|
||||
}
|
||||
return ret + iov_ret;
|
||||
}
|
||||
|
||||
int do_qtfs_conn_recv(struct qtfs_conn_var_s *pvar, bool block)
|
||||
{
|
||||
int ret = 0;
|
||||
int headlen = 0;
|
||||
struct qtreq *rsp = NULL;
|
||||
struct kvec load;
|
||||
unsigned long retrytimes = 0;
|
||||
|
||||
headlen = pvar->conn_ops->conn_recv(&pvar->conn_var, pvar->vec_recv.iov_base, QTFS_MSG_HEAD_LEN, block);
|
||||
|
||||
if (headlen <= 0) {
|
||||
return headlen;
|
||||
}
|
||||
|
||||
load.iov_base = pvar->vec_recv.iov_base + QTFS_MSG_HEAD_LEN;
|
||||
load.iov_len = pvar->vec_recv.iov_len - QTFS_MSG_HEAD_LEN;
|
||||
rsp = pvar->vec_recv.iov_base;
|
||||
|
||||
// only recv head
|
||||
if (load.iov_len == 0)
|
||||
goto end;
|
||||
|
||||
retry:
|
||||
ret = pvar->conn_ops->conn_recv(&pvar->conn_var, load.iov_base,
|
||||
(rsp->len < load.iov_len) ? rsp->len : load.iov_len, true);
|
||||
if (ret == -EAGAIN)
|
||||
goto retry;
|
||||
if (ret == -ERESTARTSYS) {
|
||||
#ifdef QTFS_CLIENT
|
||||
if (retrytimes == 0) {
|
||||
qtinfo_cntinc(QTINF_RESTART_SYS);
|
||||
qtinfo_recverrinc(rsp->type);
|
||||
}
|
||||
#endif
|
||||
retrytimes++;
|
||||
msleep(1);
|
||||
goto retry;
|
||||
}
|
||||
if (ret < 0) {
|
||||
qtfs_err("qtfs recv get invalidelen is :%d", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (ret > rsp->len) {
|
||||
qtfs_crit("recv total:%d msg len:%lu\n", ret, rsp->len);
|
||||
WARN_ON(1);
|
||||
}
|
||||
end:
|
||||
return ret + headlen;
|
||||
}
|
||||
|
||||
int qtfs_conn_recv_block(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret = 0;
|
||||
ret = qtfs_conn_sync_magic(pvar, true);
|
||||
if (ret != 0) {
|
||||
return ret;
|
||||
}
|
||||
ret = do_qtfs_conn_recv(pvar, true);
|
||||
if (ret > 0) {
|
||||
pvar->recv_valid = (ret > pvar->recv_max) ? pvar->recv_max : ret;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int qtfs_conn_recv(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret = 0;
|
||||
ret = qtfs_conn_sync_magic(pvar, true);
|
||||
if (ret != 0) {
|
||||
return ret;
|
||||
}
|
||||
ret = do_qtfs_conn_recv(pvar, false);
|
||||
if (ret <= 0) {
|
||||
msleep(1);
|
||||
} else {
|
||||
pvar->recv_valid = (ret > pvar->recv_max) ? pvar->recv_max : ret;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int qtfs_conn_var_init(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
INIT_LIST_HEAD(&pvar->lst);
|
||||
// qtfs消息为130多k,当作最大值作为合法性判断
|
||||
if (pvar->recv_max > QTFS_MSG_LEN || pvar->send_max > QTFS_MSG_LEN ||
|
||||
pvar->recv_max == 0 || pvar->recv_max == 0) {
|
||||
qtfs_err("invalid recv max:%u or invalid send max:%u",
|
||||
pvar->recv_max, pvar->send_max);
|
||||
return QTFS_ERR;
|
||||
}
|
||||
pvar->vec_recv.iov_base = kmalloc(pvar->recv_max, GFP_KERNEL);
|
||||
if (pvar->vec_recv.iov_base == NULL) {
|
||||
qtfs_err("qtfs recv kmalloc failed, len:%u.\n", pvar->recv_max);
|
||||
return QTFS_ERR;
|
||||
}
|
||||
pvar->vec_send.iov_base = kmalloc(pvar->send_max, GFP_KERNEL);
|
||||
if (pvar->vec_send.iov_base == NULL) {
|
||||
qtfs_err("qtfs send kmalloc failed, len:%u.\n", pvar->send_max);
|
||||
kfree(pvar->vec_recv.iov_base);
|
||||
pvar->vec_recv.iov_base = NULL;
|
||||
return QTFS_ERR;
|
||||
}
|
||||
pvar->vec_recv.iov_len = pvar->recv_max;
|
||||
pvar->vec_send.iov_len = 0;
|
||||
memset(pvar->vec_recv.iov_base, 0, pvar->recv_max);
|
||||
memset(pvar->vec_send.iov_base, 0, pvar->send_max);
|
||||
pvar->recv_valid = 0;
|
||||
pvar->send_valid = 0;
|
||||
qtfs_info("init pvar thread:%d recv max:%u, send max:%u", pvar->cur_threadidx, pvar->recv_max, pvar->send_max);
|
||||
return QTFS_OK;
|
||||
}
|
||||
|
||||
void qtfs_conn_var_fini(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
if (pvar->vec_recv.iov_base != NULL) {
|
||||
kfree(pvar->vec_recv.iov_base);
|
||||
pvar->vec_recv.iov_base = NULL;
|
||||
pvar->vec_recv.iov_len = 0;
|
||||
}
|
||||
if (pvar->vec_send.iov_base != NULL) {
|
||||
kfree(pvar->vec_send.iov_base);
|
||||
pvar->vec_send.iov_base = NULL;
|
||||
pvar->vec_send.iov_len = 0;
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
void qtfs_conn_msg_clear(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
memset(pvar->vec_recv.iov_base, 0, pvar->recv_valid);
|
||||
memset(pvar->vec_send.iov_base, 0, pvar->send_valid);
|
||||
pvar->recv_valid = 0;
|
||||
pvar->send_valid = 0;
|
||||
#ifdef QTFS_CLIENT
|
||||
memset(pvar->who_using, 0, QTFS_FUNCTION_LEN);
|
||||
#endif
|
||||
return;
|
||||
}
|
||||
|
||||
void *qtfs_conn_msg_buf(struct qtfs_conn_var_s *pvar, int dir)
|
||||
{
|
||||
struct qtreq *req = (dir == QTFS_SEND) ? pvar->vec_send.iov_base : pvar->vec_recv.iov_base;
|
||||
return req->data;
|
||||
}
|
||||
|
||||
// state machine
|
||||
#define QTCONN_CUR_STATE(pvar) ((pvar->state == QTCONN_INIT) ? "INIT" : \
|
||||
((pvar->state == QTCONN_CONNECTING) ? "CONNECTING" : \
|
||||
((pvar->state == QTCONN_ACTIVE) ? "ACTIVE" : "UNKNOWN")))
|
||||
|
||||
static int qtfs_sm_connecting(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret = QTERROR;
|
||||
int retry = 3;
|
||||
while (qtfs_mod_exiting == false && retry-- > 0) {
|
||||
ret = pvar->conn_ops->conn_new_connection(&pvar->conn_var, pvar->user_type);
|
||||
if (ret == 0) {
|
||||
qtfs_info("qtfs sm connecting connect to a new connection.");
|
||||
break;
|
||||
}
|
||||
msleep(100);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
int qtfs_sm_active(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
switch (pvar->state) {
|
||||
case QTCONN_ACTIVE:
|
||||
// do nothing
|
||||
break;
|
||||
case QTCONN_INIT:
|
||||
ret = qtfs_conn_init(pvar);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs sm active init failed, ret:%d.", ret);
|
||||
break;
|
||||
}
|
||||
// dont break, just enter connecting state to process
|
||||
pvar->state = QTCONN_CONNECTING;
|
||||
qtfs_info("qtfs sm active connecting, threadidx:%d",
|
||||
pvar->cur_threadidx);
|
||||
// fall-through
|
||||
|
||||
case QTCONN_CONNECTING:
|
||||
// accept(server) or connect(client)
|
||||
ret = qtfs_sm_connecting(pvar);
|
||||
if (ret == 0)
|
||||
pvar->state = QTCONN_ACTIVE;
|
||||
break;
|
||||
default:
|
||||
qtfs_err("qtfs sm active unknown state:%s.", QTCONN_CUR_STATE(pvar));
|
||||
ret = -EINVAL;
|
||||
break;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int qtfs_sm_reconnect(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret = QTOK;
|
||||
switch (pvar->state) {
|
||||
case QTCONN_INIT:
|
||||
WARN_ON(1);
|
||||
qtfs_err("qtfs sm reconnect state error!");
|
||||
ret = QTERROR;
|
||||
break;
|
||||
case QTCONN_ACTIVE:
|
||||
qtfs_conn_fini(pvar);
|
||||
ret = qtfs_conn_init(pvar);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs sm active init failed, ret:%d.", ret);
|
||||
ret = QTERROR;
|
||||
pvar->state = QTCONN_INIT;
|
||||
break;
|
||||
}
|
||||
|
||||
pvar->state = QTCONN_CONNECTING;
|
||||
qtfs_warn("qtfs sm reconnect thread:%d, state:%s.", pvar->cur_threadidx, QTCONN_CUR_STATE(pvar));
|
||||
// fall-through
|
||||
case QTCONN_CONNECTING:
|
||||
ret = qtfs_sm_connecting(pvar);
|
||||
if (ret == 0)
|
||||
pvar->state = QTCONN_ACTIVE;
|
||||
break;
|
||||
default:
|
||||
qtfs_err("qtfs sm reconnect unknown state:%s.", QTCONN_CUR_STATE(pvar));
|
||||
ret = QTERROR;
|
||||
break;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int qtfs_sm_exit(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret = QTOK;
|
||||
switch (pvar->state) {
|
||||
case QTCONN_INIT:
|
||||
// do nothing
|
||||
break;
|
||||
case QTCONN_ACTIVE:
|
||||
case QTCONN_CONNECTING:
|
||||
qtfs_conn_fini(pvar);
|
||||
#ifdef QTFS_SERVER
|
||||
pvar->state = QTCONN_CONNECTING;
|
||||
#endif
|
||||
#ifdef QTFS_CLIENT
|
||||
pvar->state = QTCONN_INIT;
|
||||
#endif
|
||||
qtfs_warn("qtfs sm exit thread:%d state:%s.", pvar->cur_threadidx, QTCONN_CUR_STATE(pvar));
|
||||
break;
|
||||
|
||||
default:
|
||||
qtfs_err("qtfs sm exit unknown state:%s.", QTCONN_CUR_STATE(pvar));
|
||||
ret = QTERROR;
|
||||
break;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int qtfs_mutex_lock_interruptible(struct mutex *lock)
|
||||
{
|
||||
int ret;
|
||||
ret = mutex_lock_interruptible(lock);
|
||||
if (ret == 0) {
|
||||
// mutex lock successed, proc lazy put
|
||||
while (1) {
|
||||
struct llist_node *toput = llist_del_first(&g_lazy_put_llst);
|
||||
struct qtfs_conn_var_s *pvar;
|
||||
if (toput == NULL)
|
||||
break;
|
||||
pvar = llist_entry(toput, struct qtfs_conn_var_s, lazy_put);
|
||||
pvar->conn_ops->conn_msg_clear(pvar);
|
||||
list_move_tail(&pvar->lst, &g_vld_lst);
|
||||
qtfs_warn("qtfs pvar lazy put idx:%d.", pvar->cur_threadidx);
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void parse_param(void)
|
||||
{
|
||||
// reserve for pcie conn type
|
||||
// default as socket type
|
||||
g_pvar_ops = &qtfs_conn_sock_pvar_ops;
|
||||
// calling conn specific parse_param
|
||||
g_pvar_ops->parse_param();
|
||||
}
|
||||
|
||||
int qtfs_conn_param_init(void)
|
||||
{
|
||||
#ifdef QTFS_CLIENT
|
||||
qtfs_fifo_pvar_cache = kmem_cache_create("qtfs_fifo_pvar",
|
||||
sizeof(struct qtfs_conn_var_s),
|
||||
0,
|
||||
(SLAB_RECLAIM_ACCOUNT | SLAB_MEM_SPREAD),
|
||||
NULL);
|
||||
if (!qtfs_fifo_pvar_cache) {
|
||||
qtfs_err("qtfs fifo pvar cache create failed.\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
#endif
|
||||
INIT_LIST_HEAD(&g_vld_lst);
|
||||
INIT_LIST_HEAD(&g_busy_lst);
|
||||
INIT_LIST_HEAD(&g_fifo_lst);
|
||||
init_llist_head(&g_lazy_put_llst);
|
||||
atomic_set(&g_qtfs_conn_num, 0);
|
||||
// parse module_param and choose specified channel
|
||||
// should set g_pvar_ops here
|
||||
parse_param();
|
||||
g_pvar_ops->param_init();
|
||||
|
||||
mutex_init(&g_param_mutex);
|
||||
mutex_init(&g_fifo_mutex);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void release_pvar(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
if (!pvar)
|
||||
return;
|
||||
|
||||
pvar->conn_ops->conn_var_fini(pvar);
|
||||
|
||||
qtfs_sm_exit(pvar);
|
||||
if (pvar->cur_threadidx < 0 || pvar->cur_threadidx >= QTFS_MAX_THREADS) {
|
||||
qtfs_err("qtfs free unknown threadidx %d", pvar->cur_threadidx);
|
||||
} else {
|
||||
qtfs_thread_var[pvar->cur_threadidx] = NULL;
|
||||
qtfs_info("qtfs free pvar idx:%d successed.", pvar->cur_threadidx);
|
||||
}
|
||||
list_del(&pvar->lst);
|
||||
kfree(pvar);
|
||||
}
|
||||
|
||||
void qtfs_conn_param_fini(void)
|
||||
{
|
||||
struct list_head *plst;
|
||||
struct list_head *n;
|
||||
int ret;
|
||||
int conn_num;
|
||||
int i;
|
||||
|
||||
#ifdef QTFS_CLIENT
|
||||
kmem_cache_destroy(qtfs_fifo_pvar_cache);
|
||||
#endif
|
||||
|
||||
ret = qtfs_mutex_lock_interruptible(&g_param_mutex);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs conn param finish mutex lock interrup failed, ret:%d.", ret);
|
||||
WARN_ON(1);
|
||||
return;
|
||||
}
|
||||
|
||||
list_for_each_safe(plst, n, &g_vld_lst) {
|
||||
release_pvar((struct qtfs_conn_var_s *)plst);
|
||||
}
|
||||
list_for_each_safe(plst, n, &g_busy_lst) {
|
||||
release_pvar((struct qtfs_conn_var_s *)plst);
|
||||
}
|
||||
|
||||
conn_num = atomic_read(&g_qtfs_conn_num);
|
||||
for (i = 0; i < conn_num; i++) {
|
||||
if (qtfs_thread_var[i] != NULL) {
|
||||
qtfs_err("qtfs param not free idx:%d holder:%s",
|
||||
qtfs_thread_var[i]->cur_threadidx,
|
||||
qtfs_thread_var[i]->who_using);
|
||||
}
|
||||
}
|
||||
mutex_unlock(&g_param_mutex);
|
||||
g_pvar_ops->param_fini();
|
||||
}
|
||||
|
||||
struct qtfs_conn_var_s *_qtfs_conn_get_param(const char *func)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = NULL;
|
||||
int ret;
|
||||
int cnt = 0;
|
||||
|
||||
if (qtfs_mod_exiting == true) {
|
||||
qtfs_warn("qtfs module is exiting, good bye!");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
retry:
|
||||
ret = qtfs_mutex_lock_interruptible(&g_param_mutex);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs conn get param mutex lock interrup failed, ret:%d.", ret);
|
||||
return NULL;
|
||||
}
|
||||
if (!list_empty(&g_vld_lst))
|
||||
pvar = list_last_entry(&g_vld_lst, struct qtfs_conn_var_s, lst);
|
||||
if (pvar != NULL) {
|
||||
list_move_tail(&pvar->lst, &g_busy_lst);
|
||||
}
|
||||
mutex_unlock(&g_param_mutex);
|
||||
|
||||
if (pvar != NULL) {
|
||||
int ret;
|
||||
if (pvar->state == QTCONN_ACTIVE && pvar->conn_ops->conn_connected(&pvar->conn_var) == false) {
|
||||
qtfs_warn("qtfs get param thread:%d disconnected, try to reconnect.", pvar->cur_threadidx);
|
||||
ret = qtfs_sm_reconnect(pvar);
|
||||
} else {
|
||||
ret = qtfs_sm_active(pvar);
|
||||
}
|
||||
if (ret != 0) {
|
||||
qtfs_conn_put_param(pvar);
|
||||
return (IS_ERR_VALUE((long)ret) ? ERR_PTR((long)ret) : NULL);
|
||||
}
|
||||
strlcpy(pvar->who_using, func, QTFS_FUNCTION_LEN);
|
||||
return pvar;
|
||||
}
|
||||
|
||||
ret = qtfs_mutex_lock_interruptible(&g_param_mutex);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs conn get param mutex lock interrup failed, ret:%d.", ret);
|
||||
return NULL;
|
||||
}
|
||||
if (atomic_read(&g_qtfs_conn_num) >= qtfs_conn_max_conn) {
|
||||
mutex_unlock(&g_param_mutex);
|
||||
cnt++;
|
||||
msleep(1);
|
||||
if (cnt < QTFS_GET_PARAM_MAX_RETRY)
|
||||
goto retry;
|
||||
qtfs_err("qtfs get param failed, the concurrency specification has reached the upper limit");
|
||||
return NULL;
|
||||
}
|
||||
pvar = kmalloc(sizeof(struct qtfs_conn_var_s), GFP_KERNEL);
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("qtfs get param kmalloc failed.\n");
|
||||
mutex_unlock(&g_param_mutex);
|
||||
return NULL;
|
||||
}
|
||||
memset(pvar, 0, sizeof(struct qtfs_conn_var_s));
|
||||
// initialize conn_pvar here
|
||||
pvar->recv_max = QTFS_MSG_LEN;
|
||||
pvar->send_max = QTFS_MSG_LEN;
|
||||
pvar->user_type = QTFS_CONN_TYPE_QTFS;
|
||||
g_pvar_ops->pvar_init(&pvar->conn_var, &pvar->conn_ops, pvar->user_type);
|
||||
if (QTFS_OK != pvar->conn_ops->conn_var_init(pvar)) {
|
||||
qtfs_err("qtfs sock var init failed.\n");
|
||||
kfree(pvar);
|
||||
mutex_unlock(&g_param_mutex);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
memcpy(pvar->who_using, func, (strlen(func) >= QTFS_FUNCTION_LEN - 1) ? (QTFS_FUNCTION_LEN - 1) : strlen(func));
|
||||
pvar->cur_threadidx = atomic_read(&g_qtfs_conn_num);
|
||||
qtfs_info("qtfs create new param, cur conn num:%d\n", atomic_read(&g_qtfs_conn_num));
|
||||
|
||||
qtfs_thread_var[pvar->cur_threadidx] = pvar;
|
||||
// add to busy list
|
||||
atomic_inc(&g_qtfs_conn_num);
|
||||
list_add(&pvar->lst, &g_busy_lst);
|
||||
|
||||
pvar->state = QTCONN_INIT;
|
||||
pvar->seq_num = 0;
|
||||
|
||||
#ifdef QTFS_CLIENT
|
||||
mutex_unlock(&g_param_mutex);
|
||||
ret = qtfs_sm_active(pvar);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs get param active connection failed, ret:%d, curstate:%s", ret, QTCONN_CUR_STATE(pvar));
|
||||
// put to vld list
|
||||
qtfs_conn_put_param(pvar);
|
||||
return (IS_ERR_VALUE((long)ret) ? ERR_PTR((long)ret) : NULL);
|
||||
}
|
||||
qtfs_thread_var[pvar->cur_threadidx] = pvar;
|
||||
#else
|
||||
if (!pvar->conn_ops->conn_inited(pvar, pvar->user_type)) {
|
||||
if ((ret = qtfs_sm_active(pvar)) != 0) {
|
||||
qtfs_err("qtfs get param active connection failed, ret:%d, curstate:%s", ret, QTCONN_CUR_STATE(pvar));
|
||||
// put to vld list
|
||||
mutex_unlock(&g_param_mutex);
|
||||
qtfs_conn_put_param(pvar);
|
||||
return (IS_ERR_VALUE((long)ret) ? ERR_PTR((long)ret) : NULL);
|
||||
}
|
||||
mutex_unlock(&g_param_mutex);
|
||||
} else {
|
||||
mutex_unlock(&g_param_mutex);
|
||||
pvar->state = QTCONN_CONNECTING;
|
||||
ret = qtfs_sm_active(pvar);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs get param active connection failed, ret:%d curstate:%s", ret, QTCONN_CUR_STATE(pvar));
|
||||
qtfs_conn_put_param(pvar);
|
||||
return (IS_ERR_VALUE((long)ret) ? ERR_PTR((long)ret) : NULL);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
qtinfo_cntinc(QTINF_ACTIV_CONN);
|
||||
|
||||
return pvar;
|
||||
}
|
||||
|
||||
struct qtfs_conn_var_s *qtfs_epoll_establish_conn(void)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = NULL;
|
||||
int ret;
|
||||
|
||||
pvar = qtfs_epoll_var;
|
||||
if (pvar) {
|
||||
if (pvar->state == QTCONN_ACTIVE && pvar->conn_ops->conn_connected(&pvar->conn_var) == false) {
|
||||
qtfs_warn("qtfs epoll get param thread:%d disconnected, try to reconnect.", pvar->cur_threadidx);
|
||||
ret = qtfs_sm_reconnect(pvar);
|
||||
} else {
|
||||
ret = qtfs_sm_active(pvar);
|
||||
}
|
||||
if (ret) {
|
||||
return NULL;
|
||||
}
|
||||
return pvar;
|
||||
}
|
||||
|
||||
pvar = kmalloc(sizeof(struct qtfs_conn_var_s), GFP_KERNEL);
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("qtfs get param kmalloc failed.\n");
|
||||
return NULL;
|
||||
}
|
||||
memset(pvar, 0, sizeof(struct qtfs_conn_var_s));
|
||||
pvar->recv_max = QTFS_EPOLL_MSG_LEN;
|
||||
pvar->send_max = QTFS_EPOLL_MSG_LEN;
|
||||
pvar->user_type = QTFS_CONN_TYPE_EPOLL;
|
||||
pvar->cur_threadidx = QTFS_EPOLL_THREADIDX;
|
||||
g_pvar_ops->pvar_init(&pvar->conn_var, &pvar->conn_ops, pvar->user_type);
|
||||
if (QTFS_OK != pvar->conn_ops->conn_var_init(pvar)) {
|
||||
qtfs_err("qtfs sock var init failed.\n");
|
||||
kfree(pvar);
|
||||
return NULL;
|
||||
}
|
||||
qtfs_epoll_var = pvar;
|
||||
pvar->state = QTCONN_INIT;
|
||||
|
||||
ret = qtfs_sm_active(pvar);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs epoll get param active new param failed, ret:%d state:%s", ret, QTCONN_CUR_STATE(pvar));
|
||||
return pvar;
|
||||
}
|
||||
|
||||
qtfs_info("qtfs create new epoll param state:%s", QTCONN_CUR_STATE(pvar));
|
||||
return pvar;
|
||||
}
|
||||
|
||||
void qtfs_conn_put_param(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret;
|
||||
if (!pvar) {
|
||||
qtfs_err("qtfs_conn_var_s is null!!");
|
||||
return;
|
||||
}
|
||||
|
||||
ret = qtfs_mutex_lock_interruptible(&g_param_mutex);
|
||||
if (ret) {
|
||||
llist_add(&pvar->lazy_put, &g_lazy_put_llst);
|
||||
qtfs_warn("qtfs conn put param add to lazy list idx:%d, ret:%d.", pvar->cur_threadidx, ret);
|
||||
return;
|
||||
}
|
||||
pvar->conn_ops->conn_msg_clear(pvar);
|
||||
list_move_tail(&pvar->lst, &g_vld_lst);
|
||||
mutex_unlock(&g_param_mutex);
|
||||
}
|
||||
|
||||
void qtfs_epoll_cut_conn(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
if (!pvar) {
|
||||
qtfs_err("qtfs_conn_var_s is null!!");
|
||||
return;
|
||||
}
|
||||
|
||||
ret = qtfs_sm_exit(pvar);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs epoll put param exit failed, ret:%d state:%s", ret, QTCONN_CUR_STATE(pvar));
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef QTFS_CLIENT
|
||||
/* fifo的机制有所不同,每一个pvar对应唯一一个fifo的访问,生命周期贯穿
|
||||
从fifo open开始到fifo close结束,在open时get param,在close时put param */
|
||||
#define QTFS_FIFO_MAGIC_SEND 0xa55aa55a
|
||||
#define QTFS_FIFO_MAGIC_RECV 0x5aa55aa5
|
||||
struct qtfs_conn_var_s *qtfs_fifo_get_param(void)
|
||||
{
|
||||
int ret;
|
||||
struct qtfs_conn_var_s *pvar = kmem_cache_alloc(qtfs_fifo_pvar_cache, GFP_KERNEL);
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("kmem cache alloc fifo cache failed.");
|
||||
return NULL;
|
||||
}
|
||||
memset(pvar, 0, sizeof(struct qtfs_conn_var_s));
|
||||
// initialize conn_pvar here
|
||||
pvar->recv_max = QTFS_FIFO_REQ_LEN;
|
||||
pvar->send_max = QTFS_FIFO_REQ_LEN;
|
||||
pvar->magic_send = QTFS_FIFO_MAGIC_SEND;
|
||||
pvar->magic_recv = QTFS_FIFO_MAGIC_RECV;
|
||||
pvar->user_type = QTFS_CONN_TYPE_FIFO;
|
||||
g_pvar_ops->pvar_init(&pvar->conn_var, &pvar->conn_ops, pvar->user_type);
|
||||
if (QTFS_OK != pvar->conn_ops->conn_var_init(pvar)) {
|
||||
qtfs_err("qtfs sock var init failed.\n");
|
||||
kmem_cache_free(qtfs_fifo_pvar_cache, pvar);
|
||||
return NULL;
|
||||
}
|
||||
pvar->state = QTCONN_INIT;
|
||||
|
||||
ret = qtfs_sm_active(pvar);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs fifo get param active new param faile, ret:%d state:%s", ret, QTCONN_CUR_STATE(pvar));
|
||||
pvar->conn_ops->conn_var_fini(pvar);
|
||||
kmem_cache_free(qtfs_fifo_pvar_cache, pvar);
|
||||
return NULL;
|
||||
}
|
||||
mutex_lock(&g_fifo_mutex);
|
||||
list_add(&pvar->lst, &g_fifo_lst);
|
||||
mutex_unlock(&g_fifo_mutex);
|
||||
qtfs_info("qtfs create new fifo param state:%s", QTCONN_CUR_STATE(pvar));
|
||||
return pvar;
|
||||
}
|
||||
|
||||
void qtfs_fifo_put_param(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
mutex_lock(&g_fifo_mutex);
|
||||
list_del(&pvar->lst);
|
||||
mutex_unlock(&g_fifo_mutex);
|
||||
qtfs_sm_exit(pvar);
|
||||
pvar->conn_ops->conn_var_fini(pvar);
|
||||
kmem_cache_free(qtfs_fifo_pvar_cache, pvar);
|
||||
return;
|
||||
}
|
||||
#endif
|
||||
|
||||
void qtfs_conn_list_cnt(void)
|
||||
{
|
||||
struct list_head *entry;
|
||||
struct qtfs_conn_var_s *pvar;
|
||||
#ifdef QTFS_CLIENT
|
||||
int ret = 0;
|
||||
ret = qtfs_mutex_lock_interruptible(&g_param_mutex);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs conn put param mutex lock interrup failed, ret:%d.", ret);
|
||||
return;
|
||||
}
|
||||
#endif
|
||||
qtfs_diag_info->pvar_busy = 0;
|
||||
qtfs_diag_info->pvar_vld = 0;
|
||||
memset(qtfs_diag_info->who_using, 0, sizeof(qtfs_diag_info->who_using));
|
||||
list_for_each(entry, &g_busy_lst) {
|
||||
qtfs_diag_info->pvar_busy++;
|
||||
pvar = (struct qtfs_conn_var_s *)entry;
|
||||
if (pvar->cur_threadidx < 0 || pvar->cur_threadidx >= QTFS_MAX_THREADS)
|
||||
continue;
|
||||
strlcpy(qtfs_diag_info->who_using[pvar->cur_threadidx],
|
||||
qtfs_thread_var[pvar->cur_threadidx]->who_using, QTFS_FUNCTION_LEN);
|
||||
}
|
||||
list_for_each(entry, &g_vld_lst)
|
||||
qtfs_diag_info->pvar_vld++;
|
||||
#ifdef QTFS_CLIENT
|
||||
mutex_unlock(&g_param_mutex);
|
||||
#endif
|
||||
}
|
||||
|
||||
module_param(qtfs_conn_max_conn, int, 0600);
|
||||
module_param_string(qtfs_log_level, qtfs_log_level, sizeof(qtfs_log_level), 0600);
|
||||
module_param_string(qtfs_conn_type, qtfs_conn_type, sizeof(qtfs_conn_type), 0600);
|
355
qtfs/qtfs_common/misc.c
Normal file
355
qtfs/qtfs_common/misc.c
Normal file
@ -0,0 +1,355 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/fs.h>
|
||||
#include <linux/miscdevice.h>
|
||||
#include <linux/mm.h>
|
||||
#include <linux/time.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/uaccess.h>
|
||||
#include <linux/mutex.h>
|
||||
|
||||
#include "comm.h"
|
||||
#include "log.h"
|
||||
#include "req.h"
|
||||
#include "conn.h"
|
||||
|
||||
struct qtfs_wl g_qtfs_wl;
|
||||
|
||||
extern struct file_operations qtfs_misc_fops;
|
||||
struct mutex qtfs_diag_info_lock;
|
||||
|
||||
static struct miscdevice qtfs_misc_dev = {
|
||||
.minor = MISC_DYNAMIC_MINOR,
|
||||
#ifndef QTFS_CLIENT
|
||||
.name = "qtfs_server",
|
||||
#else
|
||||
.name = "qtfs_client",
|
||||
#endif
|
||||
.fops = &qtfs_misc_fops,
|
||||
};
|
||||
|
||||
int qtfs_misc_register(void)
|
||||
{
|
||||
int ret = misc_register(&qtfs_misc_dev);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs misc register failed, ret:%d.", ret);
|
||||
return -EFAULT;
|
||||
}
|
||||
mutex_init(&qtfs_diag_info_lock);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void qtfs_misc_destroy(void)
|
||||
{
|
||||
misc_deregister(&qtfs_misc_dev);
|
||||
return;
|
||||
}
|
||||
|
||||
void qtfs_misc_flush_threadstate(void)
|
||||
{
|
||||
int i;
|
||||
for (i = 0; i < QTFS_MAX_THREADS; i++) {
|
||||
if (qtfs_thread_var[i] == NULL) {
|
||||
qtfs_diag_info->thread_state[i] = -1;
|
||||
continue;
|
||||
}
|
||||
qtfs_diag_info->thread_state[i] = qtfs_thread_var[i]->state;
|
||||
}
|
||||
qtfs_diag_info->epoll_state = (qtfs_epoll_var == NULL) ? -1 : qtfs_epoll_var->state;
|
||||
}
|
||||
|
||||
void qtfs_req_size(void)
|
||||
{
|
||||
qtfs_diag_info->req_size[QTFS_REQ_NULL] = sizeof(struct qtreq);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_IOCTL] = sizeof(struct qtreq_ioctl);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_STATFS] = sizeof(struct qtreq_statfs);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_MOUNT] = sizeof(struct qtreq_mount);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_OPEN] = sizeof(struct qtreq_open);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_CLOSE] = sizeof(struct qtreq_close);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_READITER] = sizeof(struct qtreq_readiter);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_WRITE] = sizeof(struct qtreq_write);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_LOOKUP] = sizeof(struct qtreq_lookup);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_READDIR] = sizeof(struct qtreq_readdir);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_MKDIR] = sizeof(struct qtreq_mkdir);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_RMDIR] = sizeof(struct qtreq_rmdir);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_GETATTR] = sizeof(struct qtreq_getattr);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_SETATTR] = sizeof(struct qtreq_setattr);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_ICREATE] = sizeof(struct qtreq_icreate);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_MKNOD] = sizeof(struct qtreq_mknod);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_UNLINK] = sizeof(struct qtreq_unlink);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_SYMLINK] = sizeof(struct qtreq_symlink);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_LINK] = sizeof(struct qtreq_link);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_GETLINK] = sizeof(struct qtreq_getlink);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_RENAME] = sizeof(struct qtreq_rename);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_XATTRLIST] = sizeof(struct qtreq_xattrlist);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_XATTRGET] = sizeof(struct qtreq_xattrget);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_SYSMOUNT] = sizeof(struct qtreq_sysmount);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_SYSUMOUNT] = sizeof(struct qtreq_sysumount);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_FIFOPOLL] = sizeof(struct qtreq_poll);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_EPOLL_CTL] = sizeof(struct qtreq_epollctl);
|
||||
qtfs_diag_info->req_size[QTFS_REQ_EPOLL_EVENT] = sizeof(struct qtreq_epollevt);
|
||||
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_NULL] = sizeof(struct qtreq);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_IOCTL] = sizeof(struct qtrsp_ioctl);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_STATFS] = sizeof(struct qtrsp_statfs);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_MOUNT] = sizeof(struct qtrsp_mount);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_OPEN] = sizeof(struct qtrsp_open);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_CLOSE] = sizeof(struct qtrsp_close);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_READITER] = sizeof(struct qtrsp_readiter);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_WRITE] = sizeof(struct qtrsp_write);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_LOOKUP] = sizeof(struct qtrsp_lookup);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_READDIR] = sizeof(struct qtrsp_readdir);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_MKDIR] = sizeof(struct qtrsp_mkdir);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_RMDIR] = sizeof(struct qtrsp_rmdir);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_GETATTR] = sizeof(struct qtrsp_getattr);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_SETATTR] = sizeof(struct qtrsp_setattr);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_ICREATE] = sizeof(struct qtrsp_icreate);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_MKNOD] = sizeof(struct qtrsp_mknod);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_UNLINK] = sizeof(struct qtrsp_unlink);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_SYMLINK] = sizeof(struct qtrsp_symlink);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_LINK] = sizeof(struct qtrsp_link);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_GETLINK] = sizeof(struct qtrsp_getlink);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_RENAME] = sizeof(struct qtrsp_rename);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_XATTRLIST] = sizeof(struct qtrsp_xattrlist);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_XATTRGET] = sizeof(struct qtrsp_xattrget);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_SYSMOUNT] = sizeof(struct qtrsp_sysmount);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_SYSUMOUNT] = sizeof(struct qtrsp_sysumount);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_FIFOPOLL] = sizeof(struct qtrsp_poll);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_EPOLL_CTL] = sizeof(struct qtrsp_epollctl);
|
||||
qtfs_diag_info->rsp_size[QTFS_REQ_EPOLL_EVENT] = sizeof(struct qtrsp_epollevt);
|
||||
}
|
||||
|
||||
void qtfs_whitelist_initset(void)
|
||||
{
|
||||
int type;
|
||||
rwlock_init(&g_qtfs_wl.rwlock);
|
||||
for (type = 0; type < QTFS_WHITELIST_MAX; type++) {
|
||||
memset(&g_qtfs_wl.cap[type], 0, sizeof(struct qtfs_wl_cap));
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
static int qtfs_whitelist_dup_check(char *tar, struct qtfs_wl_cap *cap)
|
||||
{
|
||||
int i;
|
||||
for (i = 0; i < cap->nums; i++) {
|
||||
if (strncmp(tar, cap->item[i], QTFS_PATH_MAX - 1) == 0)
|
||||
return 1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int qtfs_whitelist_add(struct qtfs_wl_item *uitem)
|
||||
{
|
||||
// uitem->type is checked
|
||||
struct qtfs_wl_cap *cap = &g_qtfs_wl.cap[uitem->type];
|
||||
write_lock(&g_qtfs_wl.rwlock);
|
||||
if (cap->nums >= QTFS_WL_MAX_NUM) {
|
||||
qtfs_err("qtfs add white list failed, nums:%u reach upper limit:%d", cap->nums, QTFS_WL_MAX_NUM);
|
||||
goto err_end;
|
||||
}
|
||||
cap->item[cap->nums] = (char *)kmalloc(uitem->len + 1, GFP_KERNEL);
|
||||
if (IS_ERR_OR_NULL(cap->item[cap->nums])) {
|
||||
qtfs_err("kmalloc error");
|
||||
goto err_end;
|
||||
}
|
||||
memset(cap->item[cap->nums], 0, uitem->len + 1);
|
||||
if (copy_from_user(cap->item[cap->nums], uitem->path, uitem->len) ||
|
||||
qtfs_whitelist_dup_check(cap->item[cap->nums], cap) == 1) {
|
||||
qtfs_err("copy from user failed or item is a duplicate, len:%u type:%u", uitem->len, uitem->type);
|
||||
kfree(cap->item[cap->nums]);
|
||||
cap->item[cap->nums] = NULL;
|
||||
goto err_end;
|
||||
}
|
||||
qtfs_info("Successed to add white list type:%u len:%u path:[%s]", uitem->type, uitem->len, cap->item[cap->nums]);
|
||||
cap->nums++;
|
||||
write_unlock(&g_qtfs_wl.rwlock);
|
||||
return 0;
|
||||
|
||||
err_end:
|
||||
write_unlock(&g_qtfs_wl.rwlock);
|
||||
return -1;
|
||||
}
|
||||
|
||||
static int qtfs_whitelist_del(struct qtfs_wl_item *uitem)
|
||||
{
|
||||
// type is checked
|
||||
struct qtfs_wl_cap *cap = &g_qtfs_wl.cap[uitem->type];
|
||||
write_lock(&g_qtfs_wl.rwlock);
|
||||
if (uitem->index >= cap->nums) {
|
||||
qtfs_err("White list del type:%u nums:%u, invalid index:%u", uitem->type, cap->nums, uitem->index);
|
||||
goto err_end;
|
||||
}
|
||||
// free target index
|
||||
kfree(cap->item[uitem->index]);
|
||||
cap->item[uitem->index] = NULL;
|
||||
if (cap->nums > 1) {
|
||||
// if nums > 1 move last one to fill the hole
|
||||
cap->item[uitem->index] = cap->item[cap->nums - 1];
|
||||
cap->item[cap->nums - 1] = NULL;
|
||||
}
|
||||
qtfs_info("white list del type:%u total nums:%u delindex:%u", uitem->type, cap->nums, uitem->index);
|
||||
|
||||
cap->nums--;
|
||||
write_unlock(&g_qtfs_wl.rwlock);
|
||||
return 0;
|
||||
err_end:
|
||||
write_unlock(&g_qtfs_wl.rwlock);
|
||||
return -1;
|
||||
}
|
||||
|
||||
static int qtfs_whitelist_get(struct qtfs_wl_item *uitem)
|
||||
{
|
||||
// type is checked
|
||||
struct qtfs_wl_cap *cap = &g_qtfs_wl.cap[uitem->type];
|
||||
int len;
|
||||
read_lock(&g_qtfs_wl.rwlock);
|
||||
if (uitem->index >= cap->nums) {
|
||||
qtfs_err("query white list invalid index:%u type:%u total nums:%u", uitem->index, uitem->type, cap->nums);
|
||||
goto err_end;
|
||||
}
|
||||
len = strlen(cap->item[uitem->index]);
|
||||
if (!access_ok(uitem->path, len)) {
|
||||
qtfs_err("white list query pointer of userspace is not valid type:%u index:%u len:%d", uitem->type, uitem->index, len);
|
||||
goto err_end;
|
||||
}
|
||||
if (copy_to_user(uitem->path, cap->item[uitem->index], len)) {
|
||||
qtfs_err("white list query copy to user failed, type:%u index:%u len:%d", uitem->type, uitem->index, len);
|
||||
goto err_end;
|
||||
}
|
||||
qtfs_info("white list query type:%u total nums:%u index:%u len:%d", uitem->type, cap->nums, uitem->index, len);
|
||||
read_unlock(&g_qtfs_wl.rwlock);
|
||||
return 0;
|
||||
|
||||
err_end:
|
||||
read_unlock(&g_qtfs_wl.rwlock);
|
||||
return -1;
|
||||
}
|
||||
|
||||
void qtfs_whitelist_clearall(void)
|
||||
{
|
||||
struct qtfs_wl_cap *cap = NULL;
|
||||
int type;
|
||||
int item;
|
||||
write_lock(&g_qtfs_wl.rwlock);
|
||||
for (type = 0; type < QTFS_WHITELIST_MAX; type++) {
|
||||
cap = &g_qtfs_wl.cap[type];
|
||||
for (item = 0; item < cap->nums; item++) {
|
||||
kfree(cap->item[item]);
|
||||
cap->item[item] = NULL;
|
||||
}
|
||||
cap->nums = 0;
|
||||
}
|
||||
write_unlock(&g_qtfs_wl.rwlock);
|
||||
return;
|
||||
}
|
||||
|
||||
void qtfs_whitelist_exit(void)
|
||||
{
|
||||
qtfs_whitelist_clearall();
|
||||
return;
|
||||
}
|
||||
|
||||
long qtfs_misc_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
|
||||
{
|
||||
long ret = QTOK;
|
||||
qtfs_info("qtfs client misc ioctl.");
|
||||
switch (cmd) {
|
||||
case QTFS_IOCTL_ALLINFO:
|
||||
mutex_lock(&qtfs_diag_info_lock);
|
||||
if (qtfs_diag_info == NULL) {
|
||||
qtfs_err("ioctl allinfo failed, qtfs_diag_info is invalid.");
|
||||
mutex_unlock(&qtfs_diag_info_lock);
|
||||
goto err_end;
|
||||
}
|
||||
qtfs_req_size();
|
||||
qtfs_diag_info->log_level = log_level;
|
||||
qtfs_misc_flush_threadstate();
|
||||
qtfs_conn_list_cnt();
|
||||
if (copy_to_user((void *)arg, qtfs_diag_info, sizeof(struct qtinfo))) {
|
||||
qtfs_err("ioctl allinfo copy to user failed.");
|
||||
mutex_unlock(&qtfs_diag_info_lock);
|
||||
goto err_end;
|
||||
}
|
||||
mutex_unlock(&qtfs_diag_info_lock);
|
||||
break;
|
||||
case QTFS_IOCTL_CLEARALL:
|
||||
mutex_lock(&qtfs_diag_info_lock);
|
||||
qtinfo_clear();
|
||||
mutex_unlock(&qtfs_diag_info_lock);
|
||||
break;
|
||||
case QTFS_IOCTL_LOGLEVEL: {
|
||||
char level_str[QTFS_LOGLEVEL_STRLEN] = {0};
|
||||
if (arg == 0 || copy_from_user(level_str, (void *)arg, QTFS_LOGLEVEL_STRLEN - 1)) {
|
||||
qtfs_err("ioctl set log level failed, arg:%lu.", arg);
|
||||
goto err_end;
|
||||
}
|
||||
ret = (long)qtfs_log_init(level_str, QTFS_LOGLEVEL_STRLEN);
|
||||
break;
|
||||
}
|
||||
case QTFS_IOCTL_EPOLL_SUPPORT:
|
||||
if (arg == 0) {
|
||||
qtfs_epoll_mode = false;
|
||||
} else {
|
||||
qtfs_epoll_mode = true;
|
||||
}
|
||||
break;
|
||||
case QTFS_IOCTL_WL_ADD: {
|
||||
struct qtfs_wl_item head;
|
||||
if (copy_from_user(&head, (void *)arg, sizeof(struct qtfs_wl_item))) {
|
||||
qtfs_err("ioctl wl add copy from user failed");
|
||||
goto err_end;
|
||||
}
|
||||
if (head.len == 0 || head.len == MAX_PATH_LEN || head.type >= QTFS_WHITELIST_MAX ||
|
||||
!access_ok(head.path, head.len)) {
|
||||
qtfs_err("ioctl wl add len:%u type:%u invalid", head.len, head.type);
|
||||
goto err_end;
|
||||
}
|
||||
if (qtfs_whitelist_add(&head) != 0) {
|
||||
qtfs_err("ioctl wl add failed!");
|
||||
goto err_end;
|
||||
}
|
||||
break;
|
||||
}
|
||||
case QTFS_IOCTL_WL_DEL:
|
||||
case QTFS_IOCTL_WL_GET: {
|
||||
struct qtfs_wl_item head;
|
||||
if (copy_from_user(&head, (void *)arg, sizeof(struct qtfs_wl_item))) {
|
||||
qtfs_err("ioctl wl del copy from user failed");
|
||||
goto err_end;
|
||||
}
|
||||
if (head.type >= QTFS_WHITELIST_MAX) {
|
||||
qtfs_err("ioctl wl del invalid type:%u", head.type);
|
||||
goto err_end;
|
||||
}
|
||||
if (cmd == QTFS_IOCTL_WL_DEL) {
|
||||
if (qtfs_whitelist_del(&head) != 0) {
|
||||
qtfs_err("ioctl wl del failed!");
|
||||
goto err_end;
|
||||
}
|
||||
} else {
|
||||
if (qtfs_whitelist_get(&head) != 0) {
|
||||
qtfs_err("ioctl wl get failed!");
|
||||
goto err_end;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
err_end:
|
||||
return QTERROR;
|
||||
}
|
332
qtfs/qtfs_common/qtfs_check.c
Normal file
332
qtfs/qtfs_common/qtfs_check.c
Normal file
@ -0,0 +1,332 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
|
||||
#include "req.h"
|
||||
#include "qtfs_check.h"
|
||||
|
||||
/*
|
||||
检查原则:
|
||||
1. 基本数据类型,据实严格判断合法范围,有数组、指针操作的注意数组越界或指针飞踩;
|
||||
2. 字符串类型,基本的判断在长度范围内要有结束符,多个字符串拼接的,每一段起始和
|
||||
结束都能被结束符分开。
|
||||
|
||||
*/
|
||||
|
||||
// string类型基本防护,在max范围内最后一个字符必须是结束符,防止越界访问
|
||||
static inline bool check_string(char *str, size_t max)
|
||||
{
|
||||
if (max == 0)
|
||||
return false;
|
||||
if (str[max - 1] != '\0')
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
static inline bool check_fd(int fd)
|
||||
{
|
||||
#define FILENO_STDERR 2
|
||||
if (fd <= FILENO_STDERR)
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
#define TOREQ (typeof(req))in;
|
||||
#define TORSP (typeof(rsp))in;
|
||||
int req_check_none(void *in)
|
||||
{
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_mount(void *in)
|
||||
{
|
||||
struct qtreq_mount *req = TOREQ;
|
||||
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_open(void *in)
|
||||
{
|
||||
struct qtreq_open *req = TOREQ;
|
||||
|
||||
// flags 和 mode如果错误syscall会报错,不会有安全风险
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_close(void *in)
|
||||
{
|
||||
struct qtreq_close *req = TOREQ;
|
||||
if (check_fd(req->fd))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_readiter(void *in)
|
||||
{
|
||||
struct qtreq_readiter *req = TOREQ;
|
||||
if (check_fd(req->fd))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_write(void *in)
|
||||
{
|
||||
struct qtreq_write *req = TOREQ;
|
||||
if (check_fd(req->d.fd) || req->d.buflen > sizeof(req->path_buf))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_lookup(void *in)
|
||||
{
|
||||
struct qtreq_lookup *req = TOREQ;
|
||||
if (check_string(req->fullname, sizeof(req->fullname)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_readdir(void *in)
|
||||
{
|
||||
struct qtreq_readdir *req = TOREQ;
|
||||
struct qtrsp_readdir *rsp = TORSP;
|
||||
if (req->count != sizeof(rsp->dirent) || check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_mkdir(void *in)
|
||||
{
|
||||
struct qtreq_mkdir *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_rmdir(void *in)
|
||||
{
|
||||
struct qtreq_rmdir *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_getattr(void *in)
|
||||
{
|
||||
struct qtreq_getattr *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_setattr(void *in)
|
||||
{
|
||||
struct qtreq_setattr *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)) || req->attr.ia_file != NULL)
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_icreate(void *in)
|
||||
{
|
||||
struct qtreq_icreate *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_mknod(void *in)
|
||||
{
|
||||
struct qtreq_mknod *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_unlink(void *in)
|
||||
{
|
||||
struct qtreq_unlink *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_symlink(void *in)
|
||||
{
|
||||
struct qtreq_symlink *req = TOREQ;
|
||||
int total_len = sizeof(req->path);
|
||||
if (req->d.newlen + req->d.oldlen >= total_len ||
|
||||
req->d.newlen == 0 || req->d.oldlen == 0)
|
||||
return QTFS_CHECK_ERR;
|
||||
if (check_string(req->path, req->d.newlen) ||
|
||||
check_string(&req->path[req->d.newlen], req->d.oldlen))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_link(void *in)
|
||||
{
|
||||
struct qtreq_link *req = TOREQ;
|
||||
int total_len = sizeof(req->path);
|
||||
if (req->d.oldlen + req->d.newlen > total_len ||
|
||||
req->d.oldlen == 0 || req->d.newlen == 0)
|
||||
return QTFS_CHECK_ERR;
|
||||
if (check_string(req->path, req->d.oldlen) ||
|
||||
check_string(&req->path[req->d.oldlen], req->d.oldlen))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_getlink(void *in)
|
||||
{
|
||||
struct qtreq_getlink *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_rename(void *in)
|
||||
{
|
||||
struct qtreq_rename *req = TOREQ;
|
||||
int total_len = sizeof(req->path);
|
||||
if (req->d.oldlen + req->d.newlen > total_len ||
|
||||
req->d.oldlen == 0 || req->d.newlen == 0)
|
||||
return QTFS_CHECK_ERR;
|
||||
if (check_string(req->path, req->d.oldlen) ||
|
||||
check_string(&req->path[req->d.oldlen], req->d.oldlen))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
int req_check_xattrlist(void *in)
|
||||
{
|
||||
struct qtreq_xattrlist *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_xattrget(void *in)
|
||||
{
|
||||
struct qtreq_xattrget *req = TOREQ;
|
||||
if (check_string(req->d.prefix_name, sizeof(req->d.prefix_name)) ||
|
||||
check_string(req->path, PATH_MAX))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_xattrset(void *in)
|
||||
{
|
||||
struct qtreq_xattrset *req = TOREQ;
|
||||
if (req->d.pathlen == 0 || req->d.namelen == 0 ||
|
||||
req->d.pathlen + req->d.namelen + req->d.valuelen > sizeof(req->buf))
|
||||
return QTFS_CHECK_ERR;
|
||||
if (check_string(req->buf, req->d.pathlen) ||
|
||||
check_string(&req->buf[req->d.pathlen], req->d.namelen) ||
|
||||
check_string(&req->buf[req->d.pathlen + req->d.namelen], req->d.valuelen))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_sysmount(void *in)
|
||||
{
|
||||
int pos = 0;
|
||||
struct qtreq_sysmount *req = TOREQ;
|
||||
if (req->d.dev_len + req->d.dir_len + req->d.type_len + req->d.data_len > sizeof(req->buf))
|
||||
return QTFS_CHECK_ERR;
|
||||
if (req->d.dir_len == 0)
|
||||
return QTFS_CHECK_ERR;
|
||||
if (check_string(req->buf, req->d.dev_len))
|
||||
return QTFS_CHECK_ERR;
|
||||
pos += req->d.dev_len;
|
||||
if (check_string(&req->buf[pos], req->d.dir_len))
|
||||
return QTFS_CHECK_ERR;
|
||||
pos += req->d.dir_len;
|
||||
if (check_string(&req->buf[pos], req->d.type_len))
|
||||
return QTFS_CHECK_ERR;
|
||||
pos += req->d.type_len;
|
||||
if (check_string(&req->buf[pos], req->d.data_len))
|
||||
return QTFS_CHECK_ERR;
|
||||
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_sysumount(void *in)
|
||||
{
|
||||
struct qtreq_sysumount *req = TOREQ;
|
||||
if (check_string(req->buf, sizeof(req->buf)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_fifopoll(void *in)
|
||||
{
|
||||
struct qtreq_poll *req = TOREQ;
|
||||
if (check_fd(req->fd))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_statfs(void *in)
|
||||
{
|
||||
struct qtreq_statfs *req = TOREQ;
|
||||
if (check_string(req->path, sizeof(req->path)))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_ioctl(void *in)
|
||||
{
|
||||
struct qtreq_ioctl *req = TOREQ;
|
||||
if (req->d.argtype != 0 && req->d.argtype != 1)
|
||||
return QTFS_CHECK_ERR;
|
||||
if (req->d.size > sizeof(req->path) || check_fd(req->d.fd))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_epoll_ctl(void *in)
|
||||
{
|
||||
struct qtreq_epollctl *req = TOREQ;
|
||||
if (check_fd(req->fd))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_llseek(void *in)
|
||||
{
|
||||
struct qtreq_llseek *req = TOREQ;
|
||||
if (check_fd(req->fd))
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_sc_kill(void *in)
|
||||
{
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_sc_sched_getaffinity(void *in)
|
||||
{
|
||||
struct qtreq_sc_sched_affinity *req = TOREQ;
|
||||
if (req->len > AFFINITY_MAX_LEN || req->len == 0)
|
||||
return QTFS_CHECK_ERR;
|
||||
return QTFS_CHECK_OK;
|
||||
}
|
||||
|
||||
int req_check_sc_sched_setaffinity(void *in)
|
||||
{
|
||||
return req_check_sc_sched_getaffinity(in);
|
||||
}
|
544
qtfs/qtfs_common/socket.c
Normal file
544
qtfs/qtfs_common/socket.c
Normal file
@ -0,0 +1,544 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
|
||||
#include <linux/kallsyms.h>
|
||||
#include <linux/tcp.h>
|
||||
#include <net/tcp.h>
|
||||
#include <linux/un.h>
|
||||
#include <linux/vm_sockets.h>
|
||||
|
||||
#include "comm.h"
|
||||
#include "conn.h"
|
||||
#include "log.h"
|
||||
#include "req.h"
|
||||
#include "symbol_wrapper.h"
|
||||
|
||||
#ifdef QTFS_TEST_MODE
|
||||
char qtfs_server_ip[20] = "127.0.0.1";
|
||||
int qtfs_server_port = 12345;
|
||||
#else
|
||||
unsigned int qtfs_server_vsock_port = 12345;
|
||||
unsigned int qtfs_server_vsock_cid = 2; // host cid in vm is always 2
|
||||
#endif
|
||||
|
||||
#ifdef QTFS_SERVER
|
||||
struct qtfs_main_sock_s {
|
||||
struct socket *sock;
|
||||
struct mutex lock;
|
||||
};
|
||||
struct qtfs_main_sock_s qtfs_server_main_sock[QTFS_CONN_TYPE_INV];
|
||||
|
||||
#endif
|
||||
|
||||
#ifdef KVER_4_19
|
||||
static inline void sock_valbool_flag(struct sock *sk, enum sock_flags bit,
|
||||
int valbool)
|
||||
{
|
||||
if (valbool)
|
||||
sock_set_flag(sk, bit);
|
||||
else
|
||||
sock_reset_flag(sk, bit);
|
||||
}
|
||||
|
||||
void sock_set_keepalive(struct sock *sk)
|
||||
{
|
||||
lock_sock(sk);
|
||||
if (sk->sk_prot->keepalive)
|
||||
sk->sk_prot->keepalive(sk, true);
|
||||
sock_valbool_flag(sk, SOCK_KEEPOPEN, true);
|
||||
release_sock(sk);
|
||||
}
|
||||
|
||||
int tcp_sock_set_keepidle_locked(struct sock *sk, int val)
|
||||
{
|
||||
struct tcp_sock *tp = tcp_sk(sk);
|
||||
|
||||
if (val < 1 || val > MAX_TCP_KEEPIDLE)
|
||||
return -EINVAL;
|
||||
|
||||
tp->keepalive_time = val * HZ;
|
||||
if (sock_flag(sk, SOCK_KEEPOPEN) &&
|
||||
!((1 << sk->sk_state) & (TCPF_CLOSE | TCPF_LISTEN))) {
|
||||
u32 elapsed = keepalive_time_elapsed(tp);
|
||||
|
||||
if (tp->keepalive_time > elapsed)
|
||||
elapsed = tp->keepalive_time - elapsed;
|
||||
else
|
||||
elapsed = 0;
|
||||
inet_csk_reset_keepalive_timer(sk, elapsed);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int tcp_sock_set_keepidle(struct sock *sk, int val)
|
||||
{
|
||||
int err;
|
||||
|
||||
lock_sock(sk);
|
||||
err = tcp_sock_set_keepidle_locked(sk, val);
|
||||
release_sock(sk);
|
||||
return err;
|
||||
}
|
||||
|
||||
int tcp_sock_set_keepintvl(struct sock *sk, int val)
|
||||
{
|
||||
if (val < 1 || val > MAX_TCP_KEEPINTVL)
|
||||
return -EINVAL;
|
||||
|
||||
lock_sock(sk);
|
||||
tcp_sk(sk)->keepalive_intvl = val * HZ;
|
||||
release_sock(sk);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int tcp_sock_set_keepcnt(struct sock *sk, int val)
|
||||
{
|
||||
if (val < 1 || val > MAX_TCP_KEEPCNT)
|
||||
return -EINVAL;
|
||||
|
||||
lock_sock(sk);
|
||||
tcp_sk(sk)->keepalive_probes = val;
|
||||
release_sock(sk);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void sock_set_reuseaddr(struct sock *sk)
|
||||
{
|
||||
lock_sock(sk);
|
||||
sk->sk_reuse = SK_CAN_REUSE;
|
||||
release_sock(sk);
|
||||
}
|
||||
#endif
|
||||
|
||||
#define QTSOCK_SET_KEEPX(sock, val) sock_set_keepalive(sock->sk); tcp_sock_set_keepcnt(sock->sk, val);\
|
||||
tcp_sock_set_keepidle(sock->sk, val); tcp_sock_set_keepintvl(sock->sk, val);
|
||||
|
||||
static int qtfs_conn_sock_recv(void *connvar, void *buf, size_t len, bool block);
|
||||
static int qtfs_conn_sock_send(void *connvar, void *buf, size_t len);
|
||||
static void qtfs_conn_sock_fini(void *connvar, qtfs_conn_type_e type);
|
||||
|
||||
void qtfs_sock_recvtimeo_set(struct socket *sock, __s64 sec, __s64 usec)
|
||||
{
|
||||
int error;
|
||||
#ifdef KVER_4_19
|
||||
struct timeval tv;
|
||||
#else
|
||||
struct __kernel_sock_timeval tv;
|
||||
sockptr_t optval = KERNEL_SOCKPTR((void *)&tv);
|
||||
#endif
|
||||
tv.tv_sec = sec;
|
||||
tv.tv_usec = usec;
|
||||
|
||||
if (sock == NULL) {
|
||||
qtfs_err("qtfs sock recvtimeo set failed, sock is invalid.");
|
||||
return;
|
||||
}
|
||||
|
||||
#ifdef KVER_4_19
|
||||
error = sock_setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO,
|
||||
(char *)&tv, sizeof(tv));
|
||||
#else
|
||||
error = sock_setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO_OLD,
|
||||
optval, sizeof(struct __kernel_sock_timeval));
|
||||
#endif
|
||||
if (error) {
|
||||
qtfs_err("qtfs param setsockopt error, ret:%d.\n", error);
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef QTFS_SERVER
|
||||
static int qtfs_conn_sock_server_accept(void *connvar, qtfs_conn_type_e type)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
int ret;
|
||||
struct socket *sock = NULL;
|
||||
if (type >= QTFS_CONN_TYPE_INV) {
|
||||
qtfs_err("invalid type:%u.", type);
|
||||
return -EFAULT;
|
||||
}
|
||||
mutex_lock(&qtfs_server_main_sock[type].lock);
|
||||
if (qtfs_server_main_sock[type].sock == NULL) {
|
||||
qtfs_err("invalid main sock, type:%u.", type);
|
||||
mutex_unlock(&qtfs_server_main_sock[type].lock);
|
||||
return -EFAULT;
|
||||
}
|
||||
sock = qtfs_server_main_sock[type].sock;
|
||||
|
||||
if (sock == NULL) {
|
||||
WARN_ON(1);
|
||||
qtfs_err("qtfs server accept failed, main sock is NULL.");
|
||||
mutex_unlock(&qtfs_server_main_sock[type].lock);
|
||||
return -EINVAL;
|
||||
}
|
||||
ret = kernel_accept(sock, &sockvar->client_sock, SOCK_NONBLOCK);
|
||||
if (ret < 0) {
|
||||
mutex_unlock(&qtfs_server_main_sock[type].lock);
|
||||
return ret;
|
||||
}
|
||||
#ifdef QTFS_TEST_MODE
|
||||
QTSOCK_SET_KEEPX(sock, 5);
|
||||
#else
|
||||
sock_set_keepalive(sock->sk);
|
||||
#endif
|
||||
|
||||
qtfs_info("qtfs accept a client connection.\n");
|
||||
qtfs_sock_recvtimeo_set(sockvar->client_sock, QTFS_SOCK_RCVTIMEO, 0);
|
||||
mutex_unlock(&qtfs_server_main_sock[type].lock);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int qtfs_conn_sock_init(void *connvar, qtfs_conn_type_e type)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
struct socket *sock;
|
||||
int ret = 0;
|
||||
int sock_family = AF_VSOCK;
|
||||
#ifdef QTFS_TEST_MODE
|
||||
struct sockaddr_in saddr;
|
||||
sock_family = AF_INET;
|
||||
saddr.sin_family = sock_family;
|
||||
saddr.sin_port = htons(sockvar->port);
|
||||
saddr.sin_addr.s_addr = in_aton(sockvar->addr);
|
||||
#else
|
||||
struct sockaddr_vm saddr;
|
||||
sock_family = AF_VSOCK;
|
||||
saddr.svm_family = sock_family;
|
||||
saddr.svm_port = sockvar->vm_port;
|
||||
saddr.svm_cid = sockvar->vm_cid;
|
||||
#endif
|
||||
if (type >= QTFS_CONN_TYPE_INV) {
|
||||
qtfs_err("invalid type in sock init:%u", type);
|
||||
ret = -EINVAL;
|
||||
goto err_end_type;
|
||||
}
|
||||
mutex_lock(&qtfs_server_main_sock[type].lock);
|
||||
if (qtfs_server_main_sock[type].sock != NULL) {
|
||||
qtfs_info("qtfs conn type:%u main sock is set, valid or out-of-date?", type);
|
||||
mutex_unlock(&qtfs_server_main_sock[type].lock);
|
||||
return 0;
|
||||
}
|
||||
|
||||
qtfs_info("qtfs sock server init enter");
|
||||
|
||||
ret = sock_create_kern(&init_net, sock_family, SOCK_STREAM, 0, &sock);
|
||||
if (ret) {
|
||||
qtfs_err("qtfs sock server init create sock failed.\n");
|
||||
goto err_end;
|
||||
}
|
||||
|
||||
sock_set_reuseaddr(sock->sk);
|
||||
#ifdef QTFS_TEST_MODE
|
||||
QTSOCK_SET_KEEPX(sock, 5);
|
||||
#else
|
||||
sock_set_keepalive(sock->sk);
|
||||
#endif
|
||||
|
||||
ret = sock->ops->bind(sock, (struct sockaddr *)&saddr, sizeof(saddr));
|
||||
if (ret < 0) {
|
||||
qtfs_err("qtfs sock server bind error: %d.\n", ret);
|
||||
goto err_end;
|
||||
}
|
||||
|
||||
ret = sock->ops->listen(sock, QTFS_SERVER_MAXCONN);
|
||||
if (ret < 0) {
|
||||
qtfs_err("qtfs sock server listen failed.\n");
|
||||
goto err_end;
|
||||
}
|
||||
|
||||
qtfs_info("qtfs socket init sock OK!");
|
||||
qtfs_server_main_sock[type].sock = sock;
|
||||
mutex_unlock(&qtfs_server_main_sock[type].lock);
|
||||
return 0;
|
||||
|
||||
err_end:
|
||||
mutex_unlock(&qtfs_server_main_sock[type].lock);
|
||||
sock_release(sock);
|
||||
err_end_type:
|
||||
return ret;
|
||||
}
|
||||
#endif
|
||||
#ifdef QTFS_CLIENT
|
||||
static int qtfs_conn_sock_client_connect(void *connvar, qtfs_conn_type_e type)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
struct socket *sock = sockvar->client_sock;
|
||||
int ret;
|
||||
#ifdef QTFS_TEST_MODE
|
||||
struct sockaddr_in saddr;
|
||||
saddr.sin_family = AF_INET;
|
||||
saddr.sin_port = htons(sockvar->port);
|
||||
saddr.sin_addr.s_addr = in_aton(sockvar->addr);
|
||||
#else
|
||||
struct sockaddr_vm saddr;
|
||||
saddr.svm_family = AF_VSOCK;
|
||||
saddr.svm_port = sockvar->vm_port;
|
||||
saddr.svm_cid = sockvar->vm_cid;
|
||||
#endif
|
||||
if (!sock) {
|
||||
qtfs_err("Invalid client sock, which is null\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = sock->ops->connect(sock, (struct sockaddr *)&saddr, sizeof(saddr), 0);
|
||||
if (ret < 0) {
|
||||
return ret;
|
||||
}
|
||||
#ifdef QTFS_TEST_MODE
|
||||
QTSOCK_SET_KEEPX(sock, 5);
|
||||
#else
|
||||
sock_set_keepalive(sock->sk);
|
||||
#endif
|
||||
|
||||
qtfs_sock_recvtimeo_set(sockvar->client_sock, QTFS_SOCK_RCVTIMEO, 0);
|
||||
return 0;
|
||||
}
|
||||
//client侧type用不上
|
||||
static int qtfs_conn_sock_init(void *connvar, qtfs_conn_type_e type)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
struct socket *sock;
|
||||
int ret;
|
||||
|
||||
#ifdef QTFS_TEST_MODE
|
||||
ret = sock_create_kern(&init_net, AF_INET, SOCK_STREAM, 0, &sock);
|
||||
#else
|
||||
ret = sock_create_kern(&init_net, AF_VSOCK, SOCK_STREAM, 0, &sock);
|
||||
#endif
|
||||
if (ret) {
|
||||
qtfs_err("qtfs sock client init create sock failed.\n");
|
||||
return -EFAULT;
|
||||
}
|
||||
#ifdef QTFS_TEST_MODE
|
||||
QTSOCK_SET_KEEPX(sock, 5);
|
||||
#else
|
||||
sock_set_keepalive(sock->sk);
|
||||
#endif
|
||||
sockvar->client_sock = sock;
|
||||
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
static int qtfs_conn_sock_recv(void *connvar, void *buf, size_t len, bool block)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
struct kvec v;
|
||||
struct msghdr msg_recv;
|
||||
memset(&msg_recv, 0, sizeof(msg_recv));
|
||||
v.iov_base = buf;
|
||||
v.iov_len = len;
|
||||
|
||||
return kernel_recvmsg(sockvar->client_sock, &msg_recv, &v, 1,
|
||||
len, (block == true) ? MSG_WAITALL : MSG_DONTWAIT);
|
||||
}
|
||||
|
||||
static int qtfs_conn_sock_recv_iter(void *connvar, struct iov_iter *iov, bool block)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
struct msghdr msg_recv;
|
||||
if (iov == NULL) {
|
||||
qtfs_err("sock recv iter is invalid");
|
||||
return -EINVAL;
|
||||
}
|
||||
memset(&msg_recv, 0, sizeof(msg_recv));
|
||||
msg_recv.msg_iter = *iov;
|
||||
return sock_recvmsg(sockvar->client_sock, &msg_recv, (block == true) ? MSG_WAITALL : MSG_DONTWAIT);
|
||||
}
|
||||
|
||||
static int qtfs_conn_sock_send_iter(void *connvar, struct iov_iter *iov)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
struct msghdr msg_send;
|
||||
if (iov == NULL) {
|
||||
qtfs_err("sock send iter is invalid");
|
||||
return -EINVAL;
|
||||
}
|
||||
memset(&msg_send, 0, sizeof(msg_send));
|
||||
msg_send.msg_iter = *iov;
|
||||
return sock_sendmsg(sockvar->client_sock, &msg_send);
|
||||
}
|
||||
|
||||
static int qtfs_conn_sock_send(void *connvar, void *buf, size_t len)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
struct kvec v;
|
||||
int ret;
|
||||
struct msghdr msg_send;
|
||||
memset(&msg_send, 0, sizeof(struct msghdr));
|
||||
|
||||
v.iov_base = buf;
|
||||
v.iov_len = len;
|
||||
|
||||
ret = kernel_sendmsg(sockvar->client_sock, &msg_send, &v, 1, len);
|
||||
if (ret < 0) {
|
||||
qtfs_err("qtfs sock send error, ret:%d.\n", ret);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void qtfs_conn_sock_fini(void *connvar, qtfs_conn_type_e type)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
if (sockvar->client_sock == NULL) {
|
||||
qtfs_err("qtfs client sock is NULL during sock_fini");
|
||||
}
|
||||
|
||||
if (sockvar->client_sock != NULL) {
|
||||
qtfs_err("qtfs conn sock finish.");
|
||||
sock_release(sockvar->client_sock);
|
||||
sockvar->client_sock = NULL;
|
||||
}
|
||||
|
||||
#ifdef QTFS_SERVER
|
||||
if (type < QTFS_CONN_TYPE_INV) {
|
||||
mutex_lock(&qtfs_server_main_sock[type].lock);
|
||||
if (qtfs_server_main_sock[type].sock != NULL) {
|
||||
sock_release(qtfs_server_main_sock[type].sock);
|
||||
qtfs_server_main_sock[type].sock = NULL;
|
||||
}
|
||||
mutex_unlock(&qtfs_server_main_sock[type].lock);
|
||||
}
|
||||
#endif
|
||||
return;
|
||||
}
|
||||
|
||||
static bool qtfs_conn_sock_connected(void *connvar)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
struct socket *sock = sockvar->client_sock;
|
||||
__u8 tcpi_state;
|
||||
if (sock == NULL)
|
||||
return false;
|
||||
tcpi_state = inet_sk_state_load(sock->sk);
|
||||
if (tcpi_state == TCP_ESTABLISHED)
|
||||
return true;
|
||||
qtfs_warn("qtfs tcpi state:%u(define:TCP_ESTABLISHED=1 is connected) disconnect!", tcpi_state);
|
||||
|
||||
return false;
|
||||
}
|
||||
#ifdef QTFS_CLIENT
|
||||
void qtfs_sock_drop_recv_buf(void *connvar)
|
||||
{
|
||||
#define TMP_STACK_LEN 64
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
int ret = 0;
|
||||
char buf[TMP_STACK_LEN];
|
||||
struct kvec vec_recv;
|
||||
struct msghdr msg_recv;
|
||||
vec_recv.iov_base = buf;
|
||||
vec_recv.iov_len = TMP_STACK_LEN;
|
||||
do {
|
||||
ret = kernel_recvmsg(sockvar->client_sock, &msg_recv, &vec_recv, 1,
|
||||
vec_recv.iov_len, MSG_DONTWAIT);
|
||||
if (ret > 0) {
|
||||
qtfs_err("drop invalid data len:%d", ret);
|
||||
}
|
||||
} while (ret > 0);
|
||||
return;
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef QTFS_SERVER
|
||||
static bool qtfs_conn_sock_inited(void *connvar, qtfs_conn_type_e type)
|
||||
{
|
||||
if (type >= QTFS_CONN_TYPE_INV) {
|
||||
qtfs_err("invalid type:%u", type);
|
||||
return false;
|
||||
}
|
||||
return qtfs_server_main_sock[type].sock != NULL;
|
||||
}
|
||||
#endif
|
||||
|
||||
int qtfs_sock_param_init(void)
|
||||
{
|
||||
#ifdef QTFS_SERVER
|
||||
int i;
|
||||
for (i = 0; i < QTFS_CONN_TYPE_INV; i++) {
|
||||
qtfs_server_main_sock[i].sock = NULL;
|
||||
mutex_init(&qtfs_server_main_sock[i].lock);
|
||||
}
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qtfs_sock_param_fini(void)
|
||||
{
|
||||
#ifdef QTFS_SERVER
|
||||
int i;
|
||||
for (i = 0; i < QTFS_CONN_TYPE_INV; i++) {
|
||||
mutex_lock(&qtfs_server_main_sock[i].lock);
|
||||
if (qtfs_server_main_sock[i].sock != NULL) {
|
||||
sock_release(qtfs_server_main_sock[i].sock);
|
||||
qtfs_server_main_sock[i].sock = NULL;
|
||||
}
|
||||
mutex_unlock(&qtfs_server_main_sock[i].lock);
|
||||
}
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct qtfs_conn_ops_s qtfs_conn_sock_ops = {
|
||||
.conn_var_init = qtfs_conn_var_init,
|
||||
.conn_var_fini = qtfs_conn_var_fini,
|
||||
.conn_msg_clear = qtfs_conn_msg_clear,
|
||||
.get_conn_msg_buf = qtfs_conn_msg_buf,
|
||||
.conn_init = qtfs_conn_sock_init,
|
||||
.conn_fini = qtfs_conn_sock_fini,
|
||||
.conn_send = qtfs_conn_sock_send,
|
||||
.conn_recv = qtfs_conn_sock_recv,
|
||||
.conn_send_iter = qtfs_conn_sock_send_iter,
|
||||
.conn_recv_iter = qtfs_conn_sock_recv_iter,
|
||||
#ifdef QTFS_SERVER
|
||||
.conn_new_connection = qtfs_conn_sock_server_accept,
|
||||
.conn_inited = qtfs_conn_sock_inited,
|
||||
#endif
|
||||
#ifdef QTFS_CLIENT
|
||||
.conn_new_connection = qtfs_conn_sock_client_connect,
|
||||
.conn_recv_buff_drop = qtfs_sock_drop_recv_buf,
|
||||
#endif
|
||||
.conn_connected = qtfs_conn_sock_connected,
|
||||
};
|
||||
|
||||
int qtfs_sock_pvar_init(void *connvar, struct qtfs_conn_ops_s **conn_ops, qtfs_conn_type_e type)
|
||||
{
|
||||
struct qtfs_sock_var_s *sockvar = (struct qtfs_sock_var_s *)connvar;
|
||||
|
||||
if (type >= QTFS_CONN_TYPE_INV) {
|
||||
qtfs_err("invalid type:%u", type);
|
||||
return -1;
|
||||
}
|
||||
|
||||
#ifdef QTFS_TEST_MODE
|
||||
// fill conn_pvar struct here
|
||||
strlcpy(sockvar->addr, qtfs_server_ip, sizeof(sockvar->addr));
|
||||
sockvar->port = qtfs_server_port + type;
|
||||
#else
|
||||
// vsock
|
||||
sockvar->vm_cid = qtfs_server_vsock_cid;
|
||||
sockvar->vm_port = qtfs_server_vsock_port + type;
|
||||
#endif
|
||||
*conn_ops = &qtfs_conn_sock_ops;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int qtfs_sock_parse_param(void)
|
||||
{
|
||||
// parse conn specific params here
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct qtfs_pvar_ops_s qtfs_conn_sock_pvar_ops = {
|
||||
.parse_param = qtfs_sock_parse_param,
|
||||
.param_init = qtfs_sock_param_init,
|
||||
.param_fini = qtfs_sock_param_fini,
|
||||
.pvar_init = qtfs_sock_pvar_init
|
||||
};
|
||||
|
||||
#ifdef QTFS_TEST_MODE
|
||||
module_param_string(qtfs_server_ip, qtfs_server_ip, sizeof(qtfs_server_ip), 0600);
|
||||
MODULE_PARM_DESC(qtfs_server_ip, "qtfs server ip");
|
||||
module_param(qtfs_server_port, int, 0600);
|
||||
#else
|
||||
module_param(qtfs_server_vsock_port, uint, 0600);
|
||||
module_param(qtfs_server_vsock_cid, uint, 0600);
|
||||
#endif
|
296
qtfs/qtfs_common/symbol_wrapper.c
Normal file
296
qtfs/qtfs_common/symbol_wrapper.c
Normal file
@ -0,0 +1,296 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/compiler_types.h>
|
||||
#include <linux/syscalls.h>
|
||||
#include <linux/trace_events.h>
|
||||
#include <linux/mount.h>
|
||||
#include <linux/file.h>
|
||||
#include <linux/kprobes.h>
|
||||
#include <linux/version.h>
|
||||
#include <asm/unistd.h>
|
||||
|
||||
#include "conn.h"
|
||||
#include "symbol_wrapper.h"
|
||||
|
||||
unsigned long *symbols_origin[SYMBOL_MAX_NUM];
|
||||
|
||||
static struct kprobe kp = {
|
||||
.symbol_name = "kallsyms_lookup_name"
|
||||
};
|
||||
struct pt_regs;
|
||||
|
||||
#ifdef __x86_64__
|
||||
#define WRAPPER_ARGS_TO_REGS1(regs) regs->di = (unsigned long)x1;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS2(regs) \
|
||||
WRAPPER_ARGS_TO_REGS1(regs)\
|
||||
regs->si = (unsigned long)x2;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS3(regs) \
|
||||
WRAPPER_ARGS_TO_REGS2(regs)\
|
||||
regs->dx = (unsigned long)x3;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS4(regs) \
|
||||
WRAPPER_ARGS_TO_REGS3(regs)\
|
||||
regs->r10 = (unsigned long)x4;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS5(regs) \
|
||||
WRAPPER_ARGS_TO_REGS4(regs)\
|
||||
regs->r8 = (unsigned long)x5;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS6(regs) \
|
||||
WRAPPER_ARGS_TO_REGS5(regs)\
|
||||
regs->r9 = (unsigned long)x6;
|
||||
#endif
|
||||
#ifdef __aarch64__
|
||||
void (*update_mapping_prot)(phys_addr_t phys, unsigned long virt, phys_addr_t size, pgprot_t prot);
|
||||
unsigned long start_rodata, end_rodata;
|
||||
|
||||
// symbols not finded in sys call table
|
||||
enum qtfs_sym_a64 {
|
||||
A64_NR_UNLINK = 0,
|
||||
A64_NR_RMDIR,
|
||||
A64_NR_EPOLL_WAIT,
|
||||
A64_NR_MAX,
|
||||
};
|
||||
unsigned long *symbols_a64[A64_NR_MAX];
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS1(regs) regs->regs[0] = (unsigned long)x1;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS2(regs)\
|
||||
WRAPPER_ARGS_TO_REGS1(regs)\
|
||||
regs->regs[1] = (unsigned long)x2;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS3(regs)\
|
||||
WRAPPER_ARGS_TO_REGS2(regs)\
|
||||
regs->regs[2] = (unsigned long)x3;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS4(regs)\
|
||||
WRAPPER_ARGS_TO_REGS3(regs)\
|
||||
regs->regs[3] = (unsigned long)x4;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS5(regs)\
|
||||
WRAPPER_ARGS_TO_REGS4(regs)\
|
||||
regs->regs[4] = (unsigned long)x5;
|
||||
|
||||
#define WRAPPER_ARGS_TO_REGS6(regs)\
|
||||
WRAPPER_ARGS_TO_REGS5(regs)\
|
||||
regs->regs[5] = (unsigned long)x6;
|
||||
#endif
|
||||
|
||||
#define KSYMS(sym, type) \
|
||||
qtfs_kern_syms.sym = (type)qtfs_kallsyms_lookup_name(#sym);
|
||||
|
||||
#define KSYMS_NULL_RETURN(sym)\
|
||||
if (sym == NULL) {\
|
||||
qtfs_err("symbol:%s not finded", #sym);\
|
||||
return -1;\
|
||||
}
|
||||
|
||||
struct qtfs_kallsyms qtfs_kern_syms;
|
||||
kallsyms_lookup_name_t qtfs_kallsyms_lookup_name;
|
||||
int qtfs_kallsyms_hack_init(void)
|
||||
{
|
||||
int ret = register_kprobe(&kp);
|
||||
if (ret < 0) {
|
||||
qtfs_err("register kprobe failed, Please confirm whether kprobe is enabled, ret:%d", ret);
|
||||
return -1;
|
||||
}
|
||||
qtfs_kallsyms_lookup_name = (kallsyms_lookup_name_t) kp.addr;
|
||||
unregister_kprobe(&kp);
|
||||
if (qtfs_kallsyms_lookup_name == NULL) {
|
||||
qtfs_err("get kallsyms function by kprobe failed.");
|
||||
return -1;
|
||||
}
|
||||
|
||||
KSYMS(sys_call_table, unsigned long **);
|
||||
KSYMS_NULL_RETURN(qtfs_kern_syms.sys_call_table);
|
||||
|
||||
KSYMS(d_absolute_path, char *(*)(const struct path *, char *, int));
|
||||
KSYMS_NULL_RETURN(qtfs_kern_syms.d_absolute_path);
|
||||
KSYMS(find_get_task_by_vpid, struct task_struct *(*)(pid_t nr));
|
||||
KSYMS_NULL_RETURN(qtfs_kern_syms.find_get_task_by_vpid);
|
||||
|
||||
#if (LINUX_VERSION_CODE < KERNEL_VERSION(5, 11, 0))
|
||||
KSYMS(__close_fd, int (*)(struct files_struct *, int));
|
||||
KSYMS_NULL_RETURN(qtfs_kern_syms.__close_fd);
|
||||
#endif
|
||||
|
||||
#ifdef __aarch64__
|
||||
update_mapping_prot = (void *)qtfs_kallsyms_lookup_name("update_mapping_prot");
|
||||
start_rodata = (unsigned long)qtfs_kallsyms_lookup_name("__start_rodata");
|
||||
end_rodata = (unsigned long)qtfs_kallsyms_lookup_name("__end_rodata");
|
||||
if (update_mapping_prot == NULL || start_rodata == NULL || end_rodata == NULL) {
|
||||
qtfs_err("failed to init memory protect handler");
|
||||
return -1;
|
||||
}
|
||||
|
||||
#pragma GCC diagnostic ignored "-Wint-conversion"
|
||||
symbols_a64[A64_NR_UNLINK] = (unsigned long)qtfs_kallsyms_lookup_name("__arm64_sys_unlink");
|
||||
KSYMS_NULL_RETURN(symbols_a64[A64_NR_UNLINK]);
|
||||
symbols_a64[A64_NR_RMDIR] = (unsigned long)qtfs_kallsyms_lookup_name("__arm64_sys_rmdir");
|
||||
KSYMS_NULL_RETURN(symbols_a64[A64_NR_RMDIR]);
|
||||
symbols_a64[A64_NR_EPOLL_WAIT] = (unsigned long)qtfs_kallsyms_lookup_name("__arm64_sys_epoll_wait");
|
||||
KSYMS_NULL_RETURN(symbols_a64[A64_NR_EPOLL_WAIT]);
|
||||
#pragma GCC diagnostic pop
|
||||
qtfs_info("finded __arm64_sys_unlink");
|
||||
qtfs_info("finded __arm64_sys_rmdir");
|
||||
qtfs_info("finded __arm64_sys_epoll_wait");
|
||||
#endif
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
__SYSCALL_DEFINEx(3, _qtfs_connect, int, fd, struct sockaddr __user *, uservaddr, int, addrlen)
|
||||
{
|
||||
return qtfs_uds_remote_connect_user(fd, uservaddr, addrlen);
|
||||
}
|
||||
|
||||
static atomic_t replace_available = ATOMIC_INIT(1);
|
||||
|
||||
int qtfs_syscall_replace_start(void)
|
||||
{
|
||||
if (!atomic_dec_and_test(&replace_available)) {
|
||||
atomic_inc(&replace_available);
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
symbols_origin[SYMBOL_SYSCALL_CONNECT] = qtfs_kern_syms.sys_call_table[__NR_connect];
|
||||
#ifdef __x86_64__
|
||||
make_rw((unsigned long)qtfs_kern_syms.sys_call_table);
|
||||
qtfs_kern_syms.sys_call_table[__NR_connect] = (unsigned long *)__x64_sys_qtfs_connect;
|
||||
make_ro((unsigned long)qtfs_kern_syms.sys_call_table);
|
||||
#endif
|
||||
|
||||
#ifdef __aarch64__
|
||||
update_mapping_prot(__pa_symbol(start_rodata), (unsigned long)start_rodata, section_size, PAGE_KERNEL);
|
||||
qtfs_kern_syms.sys_call_table[__NR_connect] = (unsigned long *)__arm64_sys_qtfs_connect;
|
||||
update_mapping_prot(__pa_symbol(start_rodata), (unsigned long)start_rodata, section_size, PAGE_KERNEL_RO);
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
void qtfs_syscall_replace_stop(void)
|
||||
{
|
||||
#ifdef __x86_64__
|
||||
make_rw((unsigned long)qtfs_kern_syms.sys_call_table);
|
||||
qtfs_kern_syms.sys_call_table[__NR_connect] = (void *)qtfs_kallsyms_lookup_name("__x64_sys_connect");
|
||||
make_ro((unsigned long)qtfs_kern_syms.sys_call_table);
|
||||
#endif
|
||||
|
||||
#ifdef __aarch64__
|
||||
update_mapping_prot(__pa_symbol(start_rodata), (unsigned long)start_rodata, section_size, PAGE_KERNEL);
|
||||
qtfs_kern_syms.sys_call_table[__NR_connect] = (void *)qtfs_kallsyms_lookup_name("__arm64_sys_connect");
|
||||
update_mapping_prot(__pa_symbol(start_rodata), (unsigned long)start_rodata, section_size, PAGE_KERNEL_RO);
|
||||
#endif
|
||||
atomic_inc(&replace_available);
|
||||
}
|
||||
|
||||
#define SYSCALL_TYPE(type) (type (*)(const struct pt_regs *))
|
||||
|
||||
#define WRAPPER_DEFINE_BYORIGIN(nargs, ret, func, nr)\
|
||||
noinline ret func\
|
||||
{\
|
||||
struct pt_regs _regs;\
|
||||
struct pt_regs *regs = &_regs;\
|
||||
ret retval;\
|
||||
WRAPPER_ARGS_TO_REGS##nargs(regs);\
|
||||
retval = (SYSCALL_TYPE(ret) symbols_origin[nr])(regs);\
|
||||
return retval;\
|
||||
}
|
||||
|
||||
#ifdef QTFS_CLIENT
|
||||
WRAPPER_DEFINE_BYORIGIN(2, long, qtfs_syscall_umount(char __user *x1, int x2), SYMBOL_SYSCALL_UMOUNT);
|
||||
WRAPPER_DEFINE_BYORIGIN(5, long, qtfs_syscall_mount(char __user *x1, char __user *x2,
|
||||
char __user *x3, unsigned long x4, void __user *x5), SYMBOL_SYSCALL_MOUNT);
|
||||
WRAPPER_DEFINE_BYORIGIN(4, long, qtfs_syscall_epoll_ctl(int x1, int x2, int x3,
|
||||
struct epoll_event __user *x4), SYMBOL_SYSCALL_EPOLL_CTL);
|
||||
#endif
|
||||
|
||||
#define WRAPPER_DEFINE(nargs, ret, func, nr)\
|
||||
noinline ret func\
|
||||
{\
|
||||
struct pt_regs _regs;\
|
||||
struct pt_regs *regs = &_regs;\
|
||||
ret retval;\
|
||||
WRAPPER_ARGS_TO_REGS##nargs(regs);\
|
||||
retval = (SYSCALL_TYPE(ret) qtfs_kern_syms.sys_call_table[nr])(regs);\
|
||||
return retval;\
|
||||
}
|
||||
|
||||
//common syscall wrapper
|
||||
WRAPPER_DEFINE_BYORIGIN(3, long, qtfs_syscall_connect(int x1, struct sockaddr __user *x2,
|
||||
int x3), SYMBOL_SYSCALL_CONNECT);
|
||||
|
||||
// only use in server syscall wrapper
|
||||
#ifdef QTFS_SERVER
|
||||
WRAPPER_DEFINE(2, long, qtfs_syscall_umount(char __user *x1, int x2), __NR_umount2);
|
||||
WRAPPER_DEFINE(5, long, qtfs_syscall_mount(char __user *x1, char __user *x2,
|
||||
char __user *x3, unsigned long x4, void __user *x5), __NR_mount);
|
||||
WRAPPER_DEFINE(4, long, qtfs_syscall_epoll_ctl(int x1, int x2, int x3,
|
||||
struct epoll_event __user *x4), __NR_epoll_ctl);
|
||||
WRAPPER_DEFINE(4, int, qtfs_syscall_readlinkat(int x1, const char __user *x2,
|
||||
char __user *x3, int x4), __NR_readlinkat);
|
||||
WRAPPER_DEFINE(5, int, qtfs_syscall_renameat2(int x1, const char __user *x2,
|
||||
int x3, const char __user *x4, unsigned int x5), __NR_renameat2);
|
||||
WRAPPER_DEFINE(3, long, qtfs_syscall_mkdirat(int x1, const char __user *x2,
|
||||
umode_t x3), __NR_mkdirat);
|
||||
WRAPPER_DEFINE(2, int, qtfs_syscall_statfs(const char __user *x1,
|
||||
struct statfs __user *x2), __NR_statfs);
|
||||
WRAPPER_DEFINE(4, int, qtfs_syscall_openat(int x1, const char __user *x2, int x3,
|
||||
umode_t x4), __NR_openat);
|
||||
WRAPPER_DEFINE(5, int, qtfs_syscall_linkat(int x1, const char __user *x2,
|
||||
int x3, const char __user *x4, int x5), __NR_linkat);
|
||||
WRAPPER_DEFINE(4, long, qtfs_syscall_mknodat(int x1, const char __user *x2, umode_t x3,
|
||||
unsigned int x4), __NR_mknodat);
|
||||
WRAPPER_DEFINE(3, off_t, qtfs_syscall_lseek(unsigned int x1, off_t x2,
|
||||
unsigned int x3), __NR_lseek);
|
||||
WRAPPER_DEFINE(2, long, qtfs_syscall_kill(pid_t x1, int x2), __NR_kill);
|
||||
WRAPPER_DEFINE(3, long, qtfs_syscall_sched_getaffinity(pid_t x1, unsigned int x2,
|
||||
unsigned long __user *x3), __NR_sched_getaffinity);
|
||||
WRAPPER_DEFINE(3, long, qtfs_syscall_sched_setaffinity(pid_t x1, unsigned int x2,
|
||||
unsigned long __user *x3), __NR_sched_setaffinity);
|
||||
WRAPPER_DEFINE(3, long, qtfs_syscall_write(unsigned int x1, const char __user *x2,
|
||||
size_t x3), __NR_write);
|
||||
WRAPPER_DEFINE(3, long, qtfs_syscall_read(unsigned int x1, char __user *x2,
|
||||
size_t x3), __NR_read);
|
||||
WRAPPER_DEFINE(3, long, qtfs_syscall_ioctl(unsigned int x1, unsigned int x2,
|
||||
unsigned long x3), __NR_ioctl);
|
||||
|
||||
#ifdef __aarch64__
|
||||
#define WRAPPER_DEFINE_A64(nargs, ret, func, nr)\
|
||||
noinline ret func\
|
||||
{\
|
||||
struct pt_regs _regs;\
|
||||
struct pt_regs *regs = &_regs;\
|
||||
ret retval;\
|
||||
WRAPPER_ARGS_TO_REGS##nargs(regs);\
|
||||
retval = (SYSCALL_TYPE(ret) symbols_a64[nr])(regs);\
|
||||
return retval;\
|
||||
}
|
||||
|
||||
// ARM64 syscall not finded in sys_call_table is defined here
|
||||
WRAPPER_DEFINE_A64(1, long, qtfs_syscall_unlink(const char __user *x1), A64_NR_UNLINK);
|
||||
WRAPPER_DEFINE_A64(4, int, qtfs_syscall_epoll_wait(int x1, struct epoll_event __user *x2,
|
||||
int x3, int x4), A64_NR_EPOLL_WAIT);
|
||||
WRAPPER_DEFINE_A64(1, long, qtfs_syscall_rmdir(const char __user *x1), A64_NR_RMDIR);
|
||||
#else
|
||||
WRAPPER_DEFINE(1, long, qtfs_syscall_unlink(const char __user *x1), __NR_unlink);
|
||||
WRAPPER_DEFINE(4, int, qtfs_syscall_epoll_wait(int x1, struct epoll_event __user *x2,
|
||||
int x3, int x4), __NR_epoll_wait);
|
||||
WRAPPER_DEFINE(1, long, qtfs_syscall_rmdir(const char __user *x1), __NR_rmdir);
|
||||
#endif
|
||||
#endif
|
||||
|
||||
|
521
qtfs/qtfs_common/user_engine.c
Normal file
521
qtfs/qtfs_common/user_engine.c
Normal file
@ -0,0 +1,521 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <sys/mman.h>
|
||||
#include <sys/time.h>
|
||||
#include <sys/stat.h>
|
||||
#include <netinet/ip.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/socket.h>
|
||||
#include <arpa/inet.h>
|
||||
#include <string.h>
|
||||
#include <pthread.h>
|
||||
#include <signal.h>
|
||||
#include <glib.h>
|
||||
#include <malloc.h>
|
||||
#include <errno.h>
|
||||
#include <sys/resource.h>
|
||||
#include <sys/prctl.h>
|
||||
#include <sys/file.h>
|
||||
#include <sys/epoll.h>
|
||||
#include <linux/vm_sockets.h>
|
||||
|
||||
#include "comm.h"
|
||||
#include "ipc/uds_main.h"
|
||||
|
||||
char wl_type_str[QTFS_WHITELIST_MAX][16] = {
|
||||
"Open",
|
||||
"Write",
|
||||
"Read",
|
||||
"Readdir",
|
||||
"Mkdir",
|
||||
"Rmdir",
|
||||
"Create",
|
||||
"Unlink",
|
||||
"Rename",
|
||||
"Setattr",
|
||||
"Setxattr",
|
||||
"Mount",
|
||||
"Kill",
|
||||
"Udsconnect"
|
||||
};
|
||||
|
||||
#define engine_out(info, ...) \
|
||||
do {\
|
||||
printf("[Engine::%s:%3d]"info"\n", __func__, __LINE__, ##__VA_ARGS__);\
|
||||
} while (0);
|
||||
|
||||
#define engine_out2(info, ...) \
|
||||
do {\
|
||||
printf("[Engine::%s:%3d]"info, __func__, __LINE__, ##__VA_ARGS__);\
|
||||
} while (0);
|
||||
|
||||
#define engine_err(info, ...) \
|
||||
do {\
|
||||
printf("[ERROR:Engine::%s:%3d]"info"\n", __func__, __LINE__, ##__VA_ARGS__);\
|
||||
} while (0);
|
||||
|
||||
#define WHITELIST_FILE "/etc/qtfs/whitelist"
|
||||
|
||||
struct engine_arg {
|
||||
int psize;
|
||||
int fd;
|
||||
int thread_idx;
|
||||
};
|
||||
|
||||
#define QTFS_USERP_SIZE QTFS_USERP_MAXSIZE
|
||||
#define QTFS_SERVER_FILE "/dev/qtfs_server"
|
||||
#define ENGINE_LOCK_ADDR "/var/run/qtfs/engine.lock"
|
||||
#define ENGINE_LOCK_FILE_DIR "/var/run/qtfs/"
|
||||
|
||||
static volatile sig_atomic_t sig_int_flag = 0;
|
||||
|
||||
static int engine_env_prepare()
|
||||
{
|
||||
DIR *dir;
|
||||
if (access(ENGINE_LOCK_ADDR, 0) == 0)
|
||||
return 0;
|
||||
if ((dir = opendir(ENGINE_LOCK_FILE_DIR)) == NULL) {
|
||||
if (mkdir(ENGINE_LOCK_FILE_DIR, 0600) < 0) {
|
||||
engine_err("mkdir %s failed.", ENGINE_LOCK_FILE_DIR);
|
||||
return -1;
|
||||
}
|
||||
} else {
|
||||
closedir(dir);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int engine_socket_lock(void)
|
||||
{
|
||||
int lock_fd = open(ENGINE_LOCK_ADDR, O_RDONLY | O_CREAT, 0600);
|
||||
if (lock_fd == -1)
|
||||
return -EINVAL;
|
||||
|
||||
return flock(lock_fd, LOCK_EX | LOCK_NB);
|
||||
}
|
||||
|
||||
int qtfs_fd;
|
||||
int engine_run = 1;
|
||||
static void qtfs_engine_userp_free(struct qtfs_server_userp_s *userp, int thread_nums)
|
||||
{
|
||||
for (int i = 0; i < thread_nums; i++) {
|
||||
if (userp[i].userp != NULL)
|
||||
free(userp[i].userp);
|
||||
if (userp[i].userp2 != NULL)
|
||||
free(userp[i].userp2);
|
||||
}
|
||||
free(userp);
|
||||
return;
|
||||
}
|
||||
|
||||
static struct qtfs_server_userp_s *qtfs_engine_thread_init(int fd, int thread_nums, int psize)
|
||||
{
|
||||
struct qtfs_server_userp_s *userp;
|
||||
userp = (struct qtfs_server_userp_s *)calloc(thread_nums, sizeof(struct qtfs_server_userp_s));
|
||||
if (userp == NULL) {
|
||||
engine_out("engine thread init malloc failed.");
|
||||
return NULL;
|
||||
}
|
||||
for (int i = 0; i < thread_nums; i++) {
|
||||
userp[i].size = psize;
|
||||
userp[i].userp = (void *)memalign(sysconf(_SC_PAGESIZE), psize);
|
||||
if (userp[i].userp == NULL) {
|
||||
engine_out("engine userp malloc failed.");
|
||||
goto rollback;
|
||||
}
|
||||
userp[i].userp2 = (void *)memalign(sysconf(_SC_PAGESIZE), psize);
|
||||
if (userp[i].userp2 == NULL) {
|
||||
engine_out("engine userp2 malloc failed.");
|
||||
goto rollback;
|
||||
}
|
||||
}
|
||||
struct qtfs_thread_init_s init_userp;
|
||||
int ret;
|
||||
init_userp.thread_nums = thread_nums;
|
||||
init_userp.userp = userp;
|
||||
ret = ioctl(fd, QTFS_IOCTL_THREAD_INIT, (unsigned long)&init_userp);
|
||||
if (ret != QTOK) {
|
||||
engine_err("Engine thread init failed reason:%s",
|
||||
(ret == EADDRINUSE) ? "Address already in use" : "userp init failed.");
|
||||
goto rollback;
|
||||
}
|
||||
return userp;
|
||||
rollback:
|
||||
qtfs_engine_userp_free(userp, thread_nums);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void *qtfs_engine_kthread(void *arg)
|
||||
{
|
||||
struct engine_arg *parg = (struct engine_arg *)arg;
|
||||
int psize = parg->psize;
|
||||
long ret;
|
||||
|
||||
while (engine_run) {
|
||||
ret = ioctl(parg->fd, QTFS_IOCTL_THREAD_RUN, 0);
|
||||
if (ret == QTEXIT) {
|
||||
engine_out("qtfs server thread:%d exit.", parg->thread_idx);
|
||||
break;
|
||||
}
|
||||
if (sig_int_flag == 1) {
|
||||
engine_out("qtfs engine recv SIGINT.");
|
||||
|
||||
if (qtfs_fd < 0) {
|
||||
engine_err("qtfs engine signal int file:%s open failed, fd:%d.", QTFS_SERVER_FILE, qtfs_fd);
|
||||
continue;
|
||||
}
|
||||
ret = ioctl(qtfs_fd, QTFS_IOCTL_EXIT, 0);
|
||||
engine_out("qtfs engine send QTFS_IOCTL_EXIT to kernel, get return value:%ld.", ret);
|
||||
engine_run = 0;
|
||||
break;
|
||||
}
|
||||
if (ret != QTOK) {
|
||||
usleep(1000);
|
||||
}
|
||||
}
|
||||
|
||||
end:
|
||||
engine_out("qtfs user engine over.");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void qtfs_signal_int(int signum)
|
||||
{
|
||||
sig_int_flag = 1;
|
||||
return;
|
||||
}
|
||||
|
||||
static void *qtfs_engine_epoll_thread(void *data)
|
||||
{
|
||||
struct engine_arg *arg = (struct engine_arg *)data;
|
||||
int fd = arg->fd;
|
||||
long ret;
|
||||
|
||||
engine_out("qtfs server epoll thread run.");
|
||||
|
||||
do {
|
||||
ret = ioctl(fd, QTFS_IOCTL_EPOLL_THREAD_INIT, NULL);
|
||||
if (ret == QTEXIT) {
|
||||
engine_out("qtfs server epoll thread init exit.");
|
||||
goto end;
|
||||
}
|
||||
} while (ret != 0 && engine_run);
|
||||
while (engine_run) {
|
||||
ret = ioctl(fd, QTFS_IOCTL_EPOLL_THREAD_RUN, NULL);
|
||||
if (ret == QTEXIT) {
|
||||
engine_out("qtfs server epoll thread exit.");
|
||||
break;
|
||||
}
|
||||
if (ret != QTOK) {
|
||||
usleep(1000);
|
||||
}
|
||||
}
|
||||
end:
|
||||
engine_out("qtfs server epoll thread exit, ret:%ld.", ret);
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int qtfs_epoll_init(int fd)
|
||||
{
|
||||
int epfd = epoll_create1(0);
|
||||
if (epfd < 0) {
|
||||
engine_err("epoll create error, ret:%d.", epfd);
|
||||
return -1;
|
||||
}
|
||||
|
||||
struct qtfs_server_epoll_s ep;
|
||||
struct epoll_event *evts;
|
||||
evts = calloc(QTFS_MAX_EPEVENTS_NUM, sizeof(struct epoll_event));
|
||||
if (evts == NULL) {
|
||||
engine_err("calloc events failed.");
|
||||
close(epfd);
|
||||
return -1;
|
||||
}
|
||||
engine_out("qtfs engine set epoll arg, fd:%d event nums:%d.", epfd, QTFS_MAX_EPEVENTS_NUM);
|
||||
ep.epfd = epfd;
|
||||
ep.event_nums = QTFS_MAX_EPEVENTS_NUM;
|
||||
ep.events = evts;
|
||||
int ret = ioctl(fd, QTFS_IOCTL_EPFDSET, &ep);
|
||||
if (ret != 0) {
|
||||
engine_err("failed to set epoll fd, ret:%d.", ret);
|
||||
close(epfd);
|
||||
return -1;
|
||||
}
|
||||
|
||||
return epfd;
|
||||
}
|
||||
|
||||
static void qtfs_whitelist_free_items(char **items, gsize count)
|
||||
{
|
||||
for (gsize j = 0; j < count; j++) {
|
||||
if (items[j])
|
||||
free(items[j]);
|
||||
}
|
||||
if (items)
|
||||
free(items);
|
||||
return;
|
||||
}
|
||||
|
||||
static int qtfs_whitelist_transfer(int fd, GKeyFile *config, int type)
|
||||
{
|
||||
gsize i, wl_count;
|
||||
int ret;
|
||||
char **items;
|
||||
struct qtfs_wl_item head;
|
||||
items = g_key_file_get_string_list(config, wl_type_str[type], "Path", &wl_count, NULL);
|
||||
if (wl_count <= 0) {
|
||||
engine_err("Can't find whitelist item %s", wl_type_str[type]);
|
||||
return -1;
|
||||
}
|
||||
if (wl_count > QTFS_WL_MAX_NUM)
|
||||
wl_count = QTFS_WL_MAX_NUM;
|
||||
|
||||
head.type = type;
|
||||
head.index = 0; // not use in add
|
||||
for(i = 0; i < wl_count; i++){
|
||||
head.len = strlen(items[i]);
|
||||
if (head.len >= QTFS_PATH_MAX) {
|
||||
engine_err("Whitelist type:%s invalid path:%s is too long(> %d)", wl_type_str[type], items[i], QTFS_PATH_MAX - 1);
|
||||
continue;
|
||||
}
|
||||
head.path = items[i];
|
||||
ret = ioctl(fd, QTFS_IOCTL_WL_ADD, &head);
|
||||
if (ret == QTERROR) {
|
||||
engine_err("Failed to add whitelist:%s type:%d, engine start failed.", items[i], type);
|
||||
goto end;
|
||||
}
|
||||
}
|
||||
engine_out("Successed to add white list items type:%s count:%d", wl_type_str[type], wl_count);
|
||||
qtfs_whitelist_free_items(items, wl_count);
|
||||
return 0;
|
||||
|
||||
end:
|
||||
qtfs_whitelist_free_items(items, wl_count);
|
||||
return -1;
|
||||
}
|
||||
|
||||
int qtfs_whitelist_init(int fd)
|
||||
{
|
||||
int mount_whitelist = 0;
|
||||
int ret, i;
|
||||
GKeyFile *config = g_key_file_new();
|
||||
g_key_file_load_from_file(config, WHITELIST_FILE, G_KEY_FILE_KEEP_COMMENTS|G_KEY_FILE_KEEP_TRANSLATIONS, NULL);
|
||||
for (i = 0; i < QTFS_WHITELIST_MAX; i++) {
|
||||
ret = qtfs_whitelist_transfer(fd, config, i);
|
||||
if (ret != 0) {
|
||||
engine_err("failed to set whitelist item %s, get error:%d", wl_type_str[i], ret);
|
||||
// failure of one whitelist type should not stop others.
|
||||
continue;
|
||||
}
|
||||
if (i == QTFS_WHITELIST_MOUNT)
|
||||
mount_whitelist = 1;
|
||||
}
|
||||
g_key_file_free(config);
|
||||
// if wl file not exist or mount not set, result is mount_whitelist == 0, stop the engine
|
||||
if (mount_whitelist == 0) {
|
||||
engine_err("Please create whitelist file and add white list items.");
|
||||
engine_err("At least one [Mount] whitelist is required for the qtfs to work normally.");
|
||||
return -1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef UDS_TEST_MODE
|
||||
static int qtfs_engine_check_port(unsigned short port, char *ip)
|
||||
#else
|
||||
static int qtfs_engine_check_port(unsigned short port, char *scid)
|
||||
#endif
|
||||
{
|
||||
#ifdef UDS_TEST_MODE
|
||||
struct sockaddr_in sin;
|
||||
if (inet_pton(AF_INET, ip, &sin.sin_addr) != 1) {
|
||||
engine_err("%s inet_pton error.", ip);
|
||||
return -1;
|
||||
}
|
||||
int sockfd = socket(AF_INET, SOCK_STREAM, 0);
|
||||
if (sockfd < 0) {
|
||||
engine_err("socket error, fd:%d", sockfd);
|
||||
return -1;
|
||||
}
|
||||
bzero(&sin, sizeof(sin));
|
||||
sin.sin_family = AF_INET;
|
||||
sin.sin_port = htons(port);
|
||||
if (bind(sockfd, (struct sockaddr *)&sin, sizeof(struct sockaddr)) < 0) {
|
||||
engine_err("ip:%s port:%u bind failed, errno:%d.", ip, port, errno);
|
||||
close(sockfd);
|
||||
return -1;
|
||||
}
|
||||
close(sockfd);
|
||||
#else
|
||||
#define DEC 10
|
||||
long cid = strtol(scid, NULL, DEC); // base 10
|
||||
if (errno == ERANGE) {
|
||||
engine_err("The cid value out of range\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (cid != -1) {
|
||||
if (cid < VMADDR_CID_HOST || cid > 0xFFFFFFFF) {
|
||||
engine_err("The cid value[%ld] was invalid\n", cid);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
struct sockaddr_vm saddr;
|
||||
memset(&saddr, 0, sizeof(saddr));
|
||||
saddr.svm_family = AF_VSOCK;
|
||||
saddr.svm_port = port;
|
||||
saddr.svm_cid = cid;
|
||||
|
||||
int sock_fd = socket(saddr.svm_family, SOCK_STREAM, 0);
|
||||
if (bind(sock_fd, (struct sockaddr *)&saddr, sizeof(saddr)) < 0) {
|
||||
engine_err("cid:%ld port:%u bind failed, errno:%d.", cid, port, errno);
|
||||
close(sock_fd);
|
||||
return -1;
|
||||
}
|
||||
close(sock_fd);
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define IS_NUMBER(c) (c >= '0' && c <= '9')
|
||||
static int qtfs_engine_env_check(char *argv[])
|
||||
{
|
||||
struct qtinfo diag;
|
||||
int ret;
|
||||
int fd;
|
||||
for (int i = 0; i < strlen(argv[1]); i ++) {
|
||||
if (!IS_NUMBER(argv[1][i])) {
|
||||
engine_err("invalid thread number :%s", argv[1]);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
if (qtfs_engine_check_port(atoi(argv[4]), argv[3]) < 0)
|
||||
goto err;
|
||||
|
||||
return 0;
|
||||
err:
|
||||
return -1;
|
||||
}
|
||||
|
||||
#define QTFS_ENGINE_FD_LIMIT 65536
|
||||
static void engine_rlimit()
|
||||
{
|
||||
struct rlimit lim;
|
||||
|
||||
getrlimit(RLIMIT_NOFILE, &lim);
|
||||
if (lim.rlim_cur >= QTFS_ENGINE_FD_LIMIT)
|
||||
return;
|
||||
engine_out("engine fd cur limit:%d, change to:%d", lim.rlim_cur, QTFS_ENGINE_FD_LIMIT);
|
||||
lim.rlim_cur = QTFS_ENGINE_FD_LIMIT;
|
||||
setrlimit(RLIMIT_NOFILE, &lim);
|
||||
return;
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
int ret = 0;
|
||||
if (argc != 7) {
|
||||
engine_out("Usage: %s <number of threads> <uds proxy thread num> <host ip> <uds proxy port> <dpu ip> <uds proxy port>.", argv[0]);
|
||||
engine_out(" Example: %s 16 1 192.168.10.10 12121 192.168.10.11 12121.", argv[0]);
|
||||
return -1;
|
||||
}
|
||||
if (engine_env_prepare() != 0 || engine_socket_lock() < 0) {
|
||||
engine_err("Engine is running.");
|
||||
return -1;
|
||||
}
|
||||
if (qtfs_engine_env_check(argv) < 0) {
|
||||
engine_err("Environment check failed, engine exit.");
|
||||
return -1;
|
||||
}
|
||||
engine_rlimit();
|
||||
int thread_nums = atoi(argv[1]);
|
||||
int fd = open(QTFS_SERVER_FILE, O_RDONLY);
|
||||
if (fd < 0) {
|
||||
engine_err("qtfs server file:%s open failed, fd:%d.", QTFS_SERVER_FILE, fd);
|
||||
return -1;
|
||||
}
|
||||
qtfs_fd = fd;
|
||||
// init epoll
|
||||
int epfd = qtfs_epoll_init(fd);
|
||||
if (epfd < 0) {
|
||||
close(fd);
|
||||
return -1;
|
||||
}
|
||||
ret = ioctl(fd, QTFS_IOCTL_EXIT, 1);
|
||||
if (ret == QTERROR) {
|
||||
goto end;
|
||||
}
|
||||
ret = qtfs_whitelist_init(fd);
|
||||
if (ret)
|
||||
goto end;
|
||||
|
||||
umask(0);
|
||||
|
||||
pthread_t texec[QTFS_MAX_THREADS];
|
||||
pthread_t tepoll;
|
||||
if (thread_nums < 0 || thread_nums > QTFS_MAX_THREADS) {
|
||||
engine_err("qtfs engine parm invalid, thread_nums:%d(must <= %d).",
|
||||
thread_nums, QTFS_MAX_THREADS);
|
||||
ret = -1;
|
||||
goto end;
|
||||
}
|
||||
signal(SIGINT, qtfs_signal_int);
|
||||
signal(SIGTERM, qtfs_signal_int);
|
||||
|
||||
struct qtfs_server_userp_s *userp = qtfs_engine_thread_init(fd, thread_nums, QTFS_USERP_SIZE);
|
||||
if (userp == NULL) {
|
||||
engine_out("qtfs engine userp init failed.");
|
||||
ret = -1;
|
||||
goto end;
|
||||
}
|
||||
struct engine_arg arg[QTFS_MAX_THREADS];
|
||||
for (int i = 0; i < thread_nums; i++) {
|
||||
arg[i].psize = QTFS_USERP_SIZE;
|
||||
arg[i].fd = fd;
|
||||
arg[i].thread_idx = i;
|
||||
(void)pthread_create(&texec[i], NULL, qtfs_engine_kthread, &arg[i]);
|
||||
}
|
||||
(void)pthread_create(&tepoll, NULL, qtfs_engine_epoll_thread, &arg[0]);
|
||||
// 必须放在这个位置,uds main里面最终也有join
|
||||
if (uds_proxy_main(6, &argv[1]) != 0) {
|
||||
engine_out("uds proxy start failed.");
|
||||
ret = -1;
|
||||
goto engine_free;
|
||||
}
|
||||
for (int i = 0; i < thread_nums; i++) {
|
||||
pthread_join(texec[i], NULL);
|
||||
engine_out("qtfs engine join thread %d.", i);
|
||||
}
|
||||
pthread_join(tepoll, NULL);
|
||||
engine_free:
|
||||
qtfs_engine_userp_free(userp, thread_nums);
|
||||
engine_out("qtfs engine join epoll thread.");
|
||||
end:
|
||||
(void)ioctl(fd, QTFS_IOCTL_EXIT, 0);
|
||||
close(epfd);
|
||||
close(fd);
|
||||
engine_out("qtfs engine over.");
|
||||
return ret;
|
||||
}
|
69
qtfs/qtfs_server/CMakeLists.txt
Normal file
69
qtfs/qtfs_server/CMakeLists.txt
Normal file
@ -0,0 +1,69 @@
|
||||
# Build Kernel modules
|
||||
set(MODULE_NAME qtfs_server)
|
||||
|
||||
## First, find the kernel build directory
|
||||
execute_process(
|
||||
COMMAND uname -r
|
||||
OUTPUT_VARIABLE KERNEL_RELEASE
|
||||
OUTPUT_STRIP_TRAILING_WHITESPACE)
|
||||
set(KBUILD_DIR /lib/modules/${KERNEL_RELEASE}/build/)
|
||||
find_file(KERNEL_MAKEFILE NAMES Makefile PATHS ${KBUILD_DIR} NO_DEFAULT_PATH)
|
||||
if (NOT KERNEL_MAKEFILE)
|
||||
message(FATAL_ERROR "There is no Makefile in ${KBUILD_DIR}!")
|
||||
endif ()
|
||||
|
||||
## Second, gather the source files
|
||||
set(COMM_SRC_FILES
|
||||
${QTFS_BASE_DIR}/qtfs_common/conn.c
|
||||
${QTFS_BASE_DIR}/qtfs_common/misc.c
|
||||
${QTFS_BASE_DIR}/qtfs_common/qtfs_check.c
|
||||
${QTFS_BASE_DIR}/qtfs_common/socket.c
|
||||
${QTFS_BASE_DIR}/qtfs_common/symbol_wrapper.c)
|
||||
|
||||
set(QTFS_SERVER_SRC_FILES
|
||||
${QTFS_BASE_DIR}/qtfs_server/fsops.c
|
||||
${QTFS_BASE_DIR}/qtfs_server/qtfs-server.c)
|
||||
|
||||
set(QTFS_SERVER_MODULE_SRC ${COMM_SRC_FILES} ${QTFS_SERVER_SRC_FILES})
|
||||
|
||||
## Third, make the src files accessible to the kernel Makefile
|
||||
set(MODULE_BUILD_DIR ${CMAKE_CURRENT_BINARY_DIR}/)
|
||||
set(QTFS_SERVER_KBUILD_FILE "obj-m := ${MODULE_NAME}.o")
|
||||
foreach (file ${QTFS_SERVER_MODULE_SRC})
|
||||
file(RELATIVE_PATH file "${CMAKE_CURRENT_SOURCE_DIR}" "${file}")
|
||||
configure_file(${file} ${file} COPYONLY)
|
||||
string(REPLACE ".c" ".o" file_obj "${file}")
|
||||
set(QTFS_SERVER_KBUILD_FILE "${QTFS_SERVER_KBUILD_FILE}\n${MODULE_NAME}-y += ${file_obj}")
|
||||
endforeach ()
|
||||
|
||||
set(QTFS_SERVER_KBUILD_FILE "${QTFS_SERVER_KBUILD_FILE}\nccflags-y += -I${QTFS_BASE_DIR}/include")
|
||||
set(QTFS_SERVER_KBUILD_FILE "${QTFS_SERVER_KBUILD_FILE}\nccflags-y += -I${QTFS_BASE_DIR}/ipc")
|
||||
set(QTFS_SERVER_KBUILD_FILE "${QTFS_SERVER_KBUILD_FILE}\nccflags-y += -I${QTFS_BASE_DIR}/qtfs_server")
|
||||
set(QTFS_SERVER_KBUILD_FILE "${QTFS_SERVER_KBUILD_FILE}\nccflags-y += -DQTFS_SERVER")
|
||||
|
||||
if (DEFINED UDS_TEST_MODE OR DEFINED QTFS_TEST_MODE)
|
||||
set(QTFS_SERVER_KBUILD_FILE "${QTFS_SERVER_KBUILD_FILE}\nccflags-y += -DQTFS_TEST_MODE")
|
||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUDS_TEST_MODE")
|
||||
endif ()
|
||||
|
||||
## Forth, generate a Kbuild file
|
||||
file(WRITE ${CMAKE_CURRENT_BINARY_DIR}/Kbuild ${QTFS_SERVER_KBUILD_FILE})
|
||||
|
||||
## Fifth, add a custom target to build the module
|
||||
set(MODULE_CMD ${CMAKE_MAKE_PROGRAM} -C ${KBUILD_DIR} M=${CMAKE_CURRENT_BINARY_DIR})
|
||||
add_custom_command(OUTPUT ${MODULE_NAME}.ko
|
||||
COMMAND ${MODULE_CMD} modules
|
||||
WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
|
||||
DEPENDS ${QTFS_SERVER_MODULE_SRC} ${CMAKE_CURRENT_BINARY_DIR}/Kbuild
|
||||
VERBATIM)
|
||||
add_custom_target(${MODULE_NAME} DEPENDS ${MODULE_NAME}.ko)
|
||||
add_custom_target(${MODULE_NAME}-clean COMMAND ${MODULE_CMD} clean)
|
||||
|
||||
# This target helps parsing C files for IDEs like CLion
|
||||
add_library(dummy-${MODULE_NAME} EXCLUDE_FROM_ALL ${QTFS_SERVER_MODULE_SRC})
|
||||
target_include_directories(dummy-${MODULE_NAME} PRIVATE
|
||||
${KBUILD_DIR}/include
|
||||
${QTFS_BASE_DIR}/include
|
||||
${QTFS_BASE_DIR}/ipc
|
||||
${QTFS_BASE_DIR}/qtfs_server)
|
||||
set_target_properties(dummy-${MODULE_NAME} PROPERTIES DEPRECATION "DO NOT BUILD THIS TARGET")
|
340
qtfs/qtfs_server/License
Normal file
340
qtfs/qtfs_server/License
Normal file
@ -0,0 +1,340 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 2, June 1991
|
||||
|
||||
Copyright (C) 1989, 1991 Free Software Foundation, Inc.
|
||||
51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The licenses for most software are designed to take away your
|
||||
freedom to share and change it. By contrast, the GNU General Public
|
||||
License is intended to guarantee your freedom to share and change free
|
||||
software--to make sure the software is free for all its users. This
|
||||
General Public License applies to most of the Free Software
|
||||
Foundation's software and to any other program whose authors commit to
|
||||
using it. (Some other Free Software Foundation software is covered by
|
||||
the GNU Library General Public License instead.) You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
this service if you wish), that you receive source code or can get it
|
||||
if you want it, that you can change the software or use pieces of it
|
||||
in new free programs; and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to make restrictions that forbid
|
||||
anyone to deny you these rights or to ask you to surrender the rights.
|
||||
These restrictions translate to certain responsibilities for you if you
|
||||
distribute copies of the software, or if you modify it.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must give the recipients all the rights that
|
||||
you have. You must make sure that they, too, receive or can get the
|
||||
source code. And you must show them these terms so they know their
|
||||
rights.
|
||||
|
||||
We protect your rights with two steps: (1) copyright the software, and
|
||||
(2) offer you this license which gives you legal permission to copy,
|
||||
distribute and/or modify the software.
|
||||
|
||||
Also, for each author's protection and ours, we want to make certain
|
||||
that everyone understands that there is no warranty for this free
|
||||
software. If the software is modified by someone else and passed on, we
|
||||
want its recipients to know that what they have is not the original, so
|
||||
that any problems introduced by others will not reflect on the original
|
||||
authors' reputations.
|
||||
|
||||
Finally, any free program is threatened constantly by software
|
||||
patents. We wish to avoid the danger that redistributors of a free
|
||||
program will individually obtain patent licenses, in effect making the
|
||||
program proprietary. To prevent this, we have made it clear that any
|
||||
patent must be licensed for everyone's free use or not licensed at all.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||
|
||||
0. This License applies to any program or other work which contains
|
||||
a notice placed by the copyright holder saying it may be distributed
|
||||
under the terms of this General Public License. The "Program", below,
|
||||
refers to any such program or work, and a "work based on the Program"
|
||||
means either the Program or any derivative work under copyright law:
|
||||
that is to say, a work containing the Program or a portion of it,
|
||||
either verbatim or with modifications and/or translated into another
|
||||
language. (Hereinafter, translation is included without limitation in
|
||||
the term "modification".) Each licensee is addressed as "you".
|
||||
|
||||
Activities other than copying, distribution and modification are not
|
||||
covered by this License; they are outside its scope. The act of
|
||||
running the Program is not restricted, and the output from the Program
|
||||
is covered only if its contents constitute a work based on the
|
||||
Program (independent of having been made by running the Program).
|
||||
Whether that is true depends on what the Program does.
|
||||
|
||||
1. You may copy and distribute verbatim copies of the Program's
|
||||
source code as you receive it, in any medium, provided that you
|
||||
conspicuously and appropriately publish on each copy an appropriate
|
||||
copyright notice and disclaimer of warranty; keep intact all the
|
||||
notices that refer to this License and to the absence of any warranty;
|
||||
and give any other recipients of the Program a copy of this License
|
||||
along with the Program.
|
||||
|
||||
You may charge a fee for the physical act of transferring a copy, and
|
||||
you may at your option offer warranty protection in exchange for a fee.
|
||||
|
||||
2. You may modify your copy or copies of the Program or any portion
|
||||
of it, thus forming a work based on the Program, and copy and
|
||||
distribute such modifications or work under the terms of Section 1
|
||||
above, provided that you also meet all of these conditions:
|
||||
|
||||
a) You must cause the modified files to carry prominent notices
|
||||
stating that you changed the files and the date of any change.
|
||||
|
||||
b) You must cause any work that you distribute or publish, that in
|
||||
whole or in part contains or is derived from the Program or any
|
||||
part thereof, to be licensed as a whole at no charge to all third
|
||||
parties under the terms of this License.
|
||||
|
||||
c) If the modified program normally reads commands interactively
|
||||
when run, you must cause it, when started running for such
|
||||
interactive use in the most ordinary way, to print or display an
|
||||
announcement including an appropriate copyright notice and a
|
||||
notice that there is no warranty (or else, saying that you provide
|
||||
a warranty) and that users may redistribute the program under
|
||||
these conditions, and telling the user how to view a copy of this
|
||||
License. (Exception: if the Program itself is interactive but
|
||||
does not normally print such an announcement, your work based on
|
||||
the Program is not required to print an announcement.)
|
||||
|
||||
These requirements apply to the modified work as a whole. If
|
||||
identifiable sections of that work are not derived from the Program,
|
||||
and can be reasonably considered independent and separate works in
|
||||
themselves, then this License, and its terms, do not apply to those
|
||||
sections when you distribute them as separate works. But when you
|
||||
distribute the same sections as part of a whole which is a work based
|
||||
on the Program, the distribution of the whole must be on the terms of
|
||||
this License, whose permissions for other licensees extend to the
|
||||
entire whole, and thus to each and every part regardless of who wrote it.
|
||||
|
||||
Thus, it is not the intent of this section to claim rights or contest
|
||||
your rights to work written entirely by you; rather, the intent is to
|
||||
exercise the right to control the distribution of derivative or
|
||||
collective works based on the Program.
|
||||
|
||||
In addition, mere aggregation of another work not based on the Program
|
||||
with the Program (or with a work based on the Program) on a volume of
|
||||
a storage or distribution medium does not bring the other work under
|
||||
the scope of this License.
|
||||
|
||||
3. You may copy and distribute the Program (or a work based on it,
|
||||
under Section 2) in object code or executable form under the terms of
|
||||
Sections 1 and 2 above provided that you also do one of the following:
|
||||
|
||||
a) Accompany it with the complete corresponding machine-readable
|
||||
source code, which must be distributed under the terms of Sections
|
||||
1 and 2 above on a medium customarily used for software interchange; or,
|
||||
|
||||
b) Accompany it with a written offer, valid for at least three
|
||||
years, to give any third party, for a charge no more than your
|
||||
cost of physically performing source distribution, a complete
|
||||
machine-readable copy of the corresponding source code, to be
|
||||
distributed under the terms of Sections 1 and 2 above on a medium
|
||||
customarily used for software interchange; or,
|
||||
|
||||
c) Accompany it with the information you received as to the offer
|
||||
to distribute corresponding source code. (This alternative is
|
||||
allowed only for noncommercial distribution and only if you
|
||||
received the program in object code or executable form with such
|
||||
an offer, in accord with Subsection b above.)
|
||||
|
||||
The source code for a work means the preferred form of the work for
|
||||
making modifications to it. For an executable work, complete source
|
||||
code means all the source code for all modules it contains, plus any
|
||||
associated interface definition files, plus the scripts used to
|
||||
control compilation and installation of the executable. However, as a
|
||||
special exception, the source code distributed need not include
|
||||
anything that is normally distributed (in either source or binary
|
||||
form) with the major components (compiler, kernel, and so on) of the
|
||||
operating system on which the executable runs, unless that component
|
||||
itself accompanies the executable.
|
||||
|
||||
If distribution of executable or object code is made by offering
|
||||
access to copy from a designated place, then offering equivalent
|
||||
access to copy the source code from the same place counts as
|
||||
distribution of the source code, even though third parties are not
|
||||
compelled to copy the source along with the object code.
|
||||
|
||||
4. You may not copy, modify, sublicense, or distribute the Program
|
||||
except as expressly provided under this License. Any attempt
|
||||
otherwise to copy, modify, sublicense or distribute the Program is
|
||||
void, and will automatically terminate your rights under this License.
|
||||
However, parties who have received copies, or rights, from you under
|
||||
this License will not have their licenses terminated so long as such
|
||||
parties remain in full compliance.
|
||||
|
||||
5. You are not required to accept this License, since you have not
|
||||
signed it. However, nothing else grants you permission to modify or
|
||||
distribute the Program or its derivative works. These actions are
|
||||
prohibited by law if you do not accept this License. Therefore, by
|
||||
modifying or distributing the Program (or any work based on the
|
||||
Program), you indicate your acceptance of this License to do so, and
|
||||
all its terms and conditions for copying, distributing or modifying
|
||||
the Program or works based on it.
|
||||
|
||||
6. Each time you redistribute the Program (or any work based on the
|
||||
Program), the recipient automatically receives a license from the
|
||||
original licensor to copy, distribute or modify the Program subject to
|
||||
these terms and conditions. You may not impose any further
|
||||
restrictions on the recipients' exercise of the rights granted herein.
|
||||
You are not responsible for enforcing compliance by third parties to
|
||||
this License.
|
||||
|
||||
7. If, as a consequence of a court judgment or allegation of patent
|
||||
infringement or for any other reason (not limited to patent issues),
|
||||
conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot
|
||||
distribute so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you
|
||||
may not distribute the Program at all. For example, if a patent
|
||||
license would not permit royalty-free redistribution of the Program by
|
||||
all those who receive copies directly or indirectly through you, then
|
||||
the only way you could satisfy both it and this License would be to
|
||||
refrain entirely from distribution of the Program.
|
||||
|
||||
If any portion of this section is held invalid or unenforceable under
|
||||
any particular circumstance, the balance of the section is intended to
|
||||
apply and the section as a whole is intended to apply in other
|
||||
circumstances.
|
||||
|
||||
It is not the purpose of this section to induce you to infringe any
|
||||
patents or other property right claims or to contest validity of any
|
||||
such claims; this section has the sole purpose of protecting the
|
||||
integrity of the free software distribution system, which is
|
||||
implemented by public license practices. Many people have made
|
||||
generous contributions to the wide range of software distributed
|
||||
through that system in reliance on consistent application of that
|
||||
system; it is up to the author/donor to decide if he or she is willing
|
||||
to distribute software through any other system and a licensee cannot
|
||||
impose that choice.
|
||||
|
||||
This section is intended to make thoroughly clear what is believed to
|
||||
be a consequence of the rest of this License.
|
||||
|
||||
8. If the distribution and/or use of the Program is restricted in
|
||||
certain countries either by patents or by copyrighted interfaces, the
|
||||
original copyright holder who places the Program under this License
|
||||
may add an explicit geographical distribution limitation excluding
|
||||
those countries, so that distribution is permitted only in or among
|
||||
countries not thus excluded. In such case, this License incorporates
|
||||
the limitation as if written in the body of this License.
|
||||
|
||||
9. The Free Software Foundation may publish revised and/or new versions
|
||||
of the General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the Program
|
||||
specifies a version number of this License which applies to it and "any
|
||||
later version", you have the option of following the terms and conditions
|
||||
either of that version or of any later version published by the Free
|
||||
Software Foundation. If the Program does not specify a version number of
|
||||
this License, you may choose any version ever published by the Free Software
|
||||
Foundation.
|
||||
|
||||
10. If you wish to incorporate parts of the Program into other free
|
||||
programs whose distribution conditions are different, write to the author
|
||||
to ask for permission. For software which is copyrighted by the Free
|
||||
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||
make exceptions for this. Our decision will be guided by the two goals
|
||||
of preserving the free status of all derivatives of our free software and
|
||||
of promoting the sharing and reuse of software generally.
|
||||
|
||||
NO WARRANTY
|
||||
|
||||
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||
REPAIR OR CORRECTION.
|
||||
|
||||
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||
POSSIBILITY OF SUCH DAMAGES.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
convey the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation; either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program; if not, write to the Free Software
|
||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program is interactive, make it output a short notice like this
|
||||
when it starts in an interactive mode:
|
||||
|
||||
Gnomovision version 69, Copyright (C) year name of author
|
||||
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, the commands you use may
|
||||
be called something other than `show w' and `show c'; they could even be
|
||||
mouse-clicks or menu items--whatever suits your program.
|
||||
|
||||
You should also get your employer (if you work as a programmer) or your
|
||||
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||
necessary. Here is a sample; alter the names:
|
||||
|
||||
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||
|
||||
<signature of Ty Coon>, 1 April 1989
|
||||
Ty Coon, President of Vice
|
||||
|
||||
This General Public License does not permit incorporating your program into
|
||||
proprietary programs. If your program is a subroutine library, you may
|
||||
consider it more useful to permit linking proprietary applications with the
|
||||
library. If this is what you want to do, use the GNU Library General
|
||||
Public License instead of this License.
|
50
qtfs/qtfs_server/Makefile
Normal file
50
qtfs/qtfs_server/Makefile
Normal file
@ -0,0 +1,50 @@
|
||||
ifdef QTFS_TEST_MODE
|
||||
ccflags-y += -I$(src)/../ -I$(src) -I$(src)/../ipc/ -I$(src)/../include/ -DQTFS_SERVER -DQTFS_TEST_MODE
|
||||
CFLAGS += -DUDS_TEST_MODE
|
||||
else
|
||||
ccflags-y += -I$(src)/../ -I$(src) -I$(src)/../ipc/ -I$(src)/../include/ -DQTFS_SERVER
|
||||
endif
|
||||
|
||||
CFLAGS += -g -O2
|
||||
CFLAGS += -fstack-protector-strong
|
||||
CFLAGS += -fPIE -pie -fPIC
|
||||
CFLAGS += -D_FORTIFY_SOURCE=2
|
||||
LDFLAGS += -Wl,-z,now
|
||||
LDFLAGS += -Wl,-z,noexecstack
|
||||
LDFLAGS += -fPIE -pie
|
||||
|
||||
KBUILD=/lib/modules/$(shell uname -r)/build/
|
||||
COMM=../qtfs_common/
|
||||
COMMO=$(COMM)/conn.o $(COMM)/misc.o $(COMM)/symbol_wrapper.o $(COMM)/socket.o $(COMM)/qtfs_check.o
|
||||
|
||||
obj-m:=qtfs_server.o
|
||||
qtfs_server-objs:=fsops.o qtfs-server.o $(COMMO)
|
||||
|
||||
DEPGLIB=-lglib-2.0 -I../ -I../include/ -I/usr/include/glib-2.0 -I/usr/lib64/glib-2.0/include
|
||||
|
||||
all: qtfs_server engine
|
||||
|
||||
qtfs_server:
|
||||
make -C $(KBUILD) M=$(PWD) modules
|
||||
@test -z $(QTFS_TEST_MODE) || echo "Important risk warning: The test mode is turned on,\
|
||||
and qtfs will expose the network port, which will bring security risks and is only for\
|
||||
testing! If you do not understand the risks, please don't use or compile again without\
|
||||
QTFS_TEST_MODE."
|
||||
|
||||
engine: uds_event.o uds_main.o user_engine.o
|
||||
gcc $(LDFLAGS) -o engine $^ -lpthread $(DEPGLIB) -I../ -I../ipc/ -DQTFS_SERVER
|
||||
|
||||
user_engine.o:
|
||||
cc $(CFLAGS) -c -o user_engine.o ../qtfs_common/user_engine.c $(DEPGLIB) -I../ -DQTFS_SERVER
|
||||
|
||||
uds_event.o:
|
||||
cc $(CFLAGS) -c -o uds_event.o ../ipc/uds_event.c -DQTFS_SERVER $(DEPGLIB)
|
||||
|
||||
uds_main.o:
|
||||
cc $(CFLAGS) -c -o uds_main.o ../ipc/uds_main.c -DQTFS_SERVER $(DEPGLIB)
|
||||
|
||||
clean:
|
||||
make -C $(KBUILD) M=$(PWD) clean
|
||||
rm -rf engine
|
||||
rm -rf ../*.o
|
||||
rm -rf $(COMMO) $(COMM).*.o.cmd
|
1734
qtfs/qtfs_server/fsops.c
Normal file
1734
qtfs/qtfs_server/fsops.c
Normal file
File diff suppressed because it is too large
Load Diff
25
qtfs/qtfs_server/fsops.h
Normal file
25
qtfs/qtfs_server/fsops.h
Normal file
@ -0,0 +1,25 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_SERVER_FSOPS_H__
|
||||
#define __QTFS_SERVER_FSOPS_H__
|
||||
|
||||
struct qtfs_getdents {
|
||||
struct dir_context ctx;
|
||||
int vldcnt;
|
||||
struct qtfs_dirent64 * dir;
|
||||
int prev_reclen;
|
||||
int count;
|
||||
};
|
||||
|
||||
#endif
|
440
qtfs/qtfs_server/qtfs-server.c
Normal file
440
qtfs/qtfs_server/qtfs-server.c
Normal file
@ -0,0 +1,440 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/init.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/kthread.h>
|
||||
#include <linux/sched.h>
|
||||
#include <linux/socket.h>
|
||||
|
||||
#include "conn.h"
|
||||
#include "qtfs-server.h"
|
||||
#include "comm.h"
|
||||
#include "log.h"
|
||||
#include "req.h"
|
||||
#include "symbol_wrapper.h"
|
||||
|
||||
#define QTFS_EPOLL_TIMEO 1000 // unit ms
|
||||
#define QTFS_EPOLL_RETRY_INTVL 500 // unit ms
|
||||
|
||||
int qtfs_server_thread_run = 1;
|
||||
DEFINE_RWLOCK(g_userp_rwlock);
|
||||
struct qtserver_fd_bitmap qtfs_fd_bitmap;
|
||||
|
||||
long qtfs_server_misc_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
|
||||
struct file_operations qtfs_misc_fops = {
|
||||
.owner=THIS_MODULE,
|
||||
.unlocked_ioctl = qtfs_server_misc_ioctl,
|
||||
};
|
||||
|
||||
struct qtfs_server_epoll_s qtfs_epoll = {
|
||||
.epfd = -1,
|
||||
.event_nums = 0,
|
||||
.events = NULL,
|
||||
.kevents = NULL,
|
||||
};
|
||||
rwlock_t qtfs_epoll_rwlock;
|
||||
|
||||
void qtfs_server_epoll_exit(void)
|
||||
{
|
||||
write_lock(&qtfs_epoll_rwlock);
|
||||
if (qtfs_epoll.kevents == NULL) {
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
return;
|
||||
}
|
||||
kfree(qtfs_epoll.kevents);
|
||||
qtfs_epoll.kevents = NULL;
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
return;
|
||||
}
|
||||
|
||||
long qtfs_server_epoll_thread(struct qtfs_conn_var_s *pvar)
|
||||
{
|
||||
int n;
|
||||
struct qtreq_epollevt *req;
|
||||
struct qtrsp_epollevt *rsp;
|
||||
struct qtreq *head;
|
||||
int sendlen;
|
||||
int ret = 0;
|
||||
int i = 0;
|
||||
|
||||
if (qtfs_epoll.epfd == -1) {
|
||||
qtfs_err("qtfs epoll wait error, epfd is invalid.");
|
||||
return QTERROR;
|
||||
}
|
||||
if (false == pvar->conn_ops->conn_connected(&pvar->conn_var)) {
|
||||
qtfs_warn("qtfs epoll thread disconnected, now try to reconnect.");
|
||||
ret = qtfs_sm_reconnect(pvar);
|
||||
} else {
|
||||
ret = qtfs_sm_active(pvar);
|
||||
}
|
||||
if (ret != QTOK) {
|
||||
qtfs_err("qtfs epoll thread connect state error, can't work.");
|
||||
msleep(QTFS_EPOLL_RETRY_INTVL);
|
||||
return QTERROR;
|
||||
}
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
rsp = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_RECV);
|
||||
head = pvar->vec_send.iov_base;
|
||||
do {
|
||||
n = qtfs_syscall_epoll_wait(qtfs_epoll.epfd, qtfs_epoll.events, qtfs_epoll.event_nums, 0);
|
||||
if (n == 0) {
|
||||
msleep(1);
|
||||
break;
|
||||
}
|
||||
if (n < 0 || n > QTFS_MAX_EPEVENTS_NUM) {
|
||||
msleep(QTFS_EPOLL_RETRY_INTVL);
|
||||
qtfs_err("epoll get new events number failed:%d ", n);
|
||||
break;
|
||||
}
|
||||
qtfs_info(">>epoll get new events number:%d.", n);
|
||||
if (copy_from_user(qtfs_epoll.kevents, qtfs_epoll.events, sizeof(struct epoll_event) * n)) {
|
||||
qtfs_err("qtfs copy epoll events failed, events lost.");
|
||||
WARN_ON(1);
|
||||
break;
|
||||
}
|
||||
for (i = 0; i < n; i++) {
|
||||
req->events[i].data = qtfs_epoll.kevents[i].data;
|
||||
req->events[i].events = qtfs_epoll.kevents[i].events;
|
||||
qtfs_info("epoll thread head req:%lx.", (unsigned long)req);
|
||||
}
|
||||
req->event_nums = n;
|
||||
sendlen = sizeof(struct qtreq_epollevt) - sizeof(req->events) + n * sizeof(struct qtreq_epoll_event);
|
||||
if (sendlen > QTFS_REQ_MAX_LEN) {
|
||||
qtfs_err("qtfs epoll events size(%d) larger than qtfs message size.", sendlen);
|
||||
WARN_ON(1);
|
||||
break;
|
||||
}
|
||||
pvar->vec_send.iov_len = QTFS_MSG_HEAD_LEN + sendlen;
|
||||
head->len = sendlen;
|
||||
head->type = QTFS_REQ_EPOLL_EVENT;
|
||||
ret = qtfs_conn_send(pvar);
|
||||
qtfs_info("qtfs send msg conn: sendlen:%lu ret:%d.",
|
||||
(unsigned long)pvar->vec_send.iov_len, ret);
|
||||
if (ret == -EPIPE) {
|
||||
qtfs_err("epoll wait send events failed get EPIPE, just wait new connection.");
|
||||
qtfs_sm_reconnect(pvar);
|
||||
break;
|
||||
}
|
||||
if (ret < 0) {
|
||||
qtfs_err("epoll wait send events failed, ret:%d.", ret);
|
||||
WARN_ON(1);
|
||||
}
|
||||
retry:
|
||||
ret = qtfs_conn_recv_block(pvar);
|
||||
if (ret == -EAGAIN) {
|
||||
if (qtfs_server_thread_run == 0) {
|
||||
qtfs_warn("qtfs module exiting, goodbye!");
|
||||
return QTEXIT;
|
||||
}
|
||||
goto retry;
|
||||
}
|
||||
if (ret == -EPIPE) {
|
||||
qtfs_err("epoll recv events failed get EPIPE, just wait new connection.");
|
||||
qtfs_sm_reconnect(pvar);
|
||||
break;
|
||||
}
|
||||
}while (0);
|
||||
|
||||
return (ret < 0) ? QTERROR : QTOK;
|
||||
}
|
||||
|
||||
long qtfs_server_epoll_init(void)
|
||||
{
|
||||
struct qtfs_conn_var_s *pvar = NULL;
|
||||
struct qtreq_epollevt *req;
|
||||
|
||||
pvar = qtfs_epoll_establish_conn();
|
||||
if (pvar == NULL) {
|
||||
return QTERROR;
|
||||
}
|
||||
qtfs_epoll_var = pvar;
|
||||
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
qtfs_info("qtfs epoll events req size:%lu, events size:%lu, struct:%lu.",
|
||||
sizeof(struct qtreq_epollevt), sizeof(req->events),
|
||||
sizeof(struct qtreq_epoll_event));
|
||||
qtfs_info("qtfs epoll wait thread, epfd:%d nums:%d.",
|
||||
qtfs_epoll.epfd, qtfs_epoll.event_nums);
|
||||
|
||||
return QTOK;
|
||||
}
|
||||
|
||||
int qtfs_server_fd_bitmap_init(void)
|
||||
{
|
||||
struct rlimit fd_rlim;
|
||||
if (qtfs_fd_bitmap.bitmap != NULL) {
|
||||
qtfs_info("free old bitmap");
|
||||
kfree(qtfs_fd_bitmap.bitmap);
|
||||
qtfs_fd_bitmap.bitmap = NULL;
|
||||
qtfs_fd_bitmap.nbits = 0;
|
||||
}
|
||||
// fd_rlim is get from current task struct, can be trusted
|
||||
fd_rlim = current->signal->rlim[RLIMIT_NOFILE];
|
||||
qtfs_info("task rlimit cur:%lu max:%lu", fd_rlim.rlim_cur, fd_rlim.rlim_max);
|
||||
qtfs_fd_bitmap.bitmap = (unsigned long *)kmalloc(BITS_TO_BYTES(fd_rlim.rlim_cur), GFP_KERNEL);
|
||||
if (qtfs_fd_bitmap.bitmap == NULL) {
|
||||
qtfs_err("kmalloc len:%lu failed.", BITS_TO_BYTES(fd_rlim.rlim_cur));
|
||||
return -1;
|
||||
}
|
||||
qtfs_fd_bitmap.nbits = fd_rlim.rlim_cur;
|
||||
bitmap_zero(qtfs_fd_bitmap.bitmap, qtfs_fd_bitmap.nbits);
|
||||
return 0;
|
||||
}
|
||||
|
||||
long qtfs_server_misc_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
|
||||
{
|
||||
int i;
|
||||
long ret = 0;
|
||||
struct qtfs_conn_var_s *pvar;
|
||||
struct qtfs_thread_init_s init_userp;
|
||||
switch (cmd) {
|
||||
case QTFS_IOCTL_THREAD_INIT:
|
||||
pvar = qtfs_conn_get_param_errcode();
|
||||
if (IS_ERR_OR_NULL(pvar)) {
|
||||
qtfs_err("init pvar get failed, pvar:%ld", (long)pvar);
|
||||
if (PTR_ERR(pvar) == -EADDRINUSE)
|
||||
return EADDRINUSE;
|
||||
} else {
|
||||
qtfs_conn_put_param(pvar);
|
||||
}
|
||||
if (!write_trylock(&g_userp_rwlock)) {
|
||||
qtfs_err("try lock userps failed.");
|
||||
return QTERROR;
|
||||
}
|
||||
if (qtfs_server_fd_bitmap_init() < 0) {
|
||||
qtfs_err("fd bitmap init failed.");
|
||||
write_unlock(&g_userp_rwlock);
|
||||
return QTERROR;
|
||||
}
|
||||
if (copy_from_user(&init_userp, (void __user *)arg, sizeof(struct qtfs_thread_init_s))) {
|
||||
qtfs_err("qtfs ioctl thread init copy from user failed.");
|
||||
write_unlock(&g_userp_rwlock);
|
||||
return QTERROR;
|
||||
}
|
||||
if (qtfs_userps == NULL || init_userp.thread_nums > QTFS_MAX_THREADS || init_userp.thread_nums == 0) {
|
||||
qtfs_err("qtfs ioctl thread init userps invalid thread nums:%d.", init_userp.thread_nums);
|
||||
write_unlock(&g_userp_rwlock);
|
||||
return QTERROR;
|
||||
}
|
||||
memset(qtfs_userps, 0, QTFS_MAX_THREADS * sizeof(struct qtfs_server_userp_s));
|
||||
if (init_userp.thread_nums > QTFS_MAX_THREADS) {
|
||||
qtfs_err("qtfs ioctl thread init invalid input thread_num:%d", init_userp.thread_nums);
|
||||
write_unlock(&g_userp_rwlock);
|
||||
return QTERROR;
|
||||
}
|
||||
if (copy_from_user(qtfs_userps, (void __user *)init_userp.userp,
|
||||
init_userp.thread_nums * sizeof(struct qtfs_server_userp_s))) {
|
||||
qtfs_err("qtfs ioctl thread init copy from userp failed.");
|
||||
write_unlock(&g_userp_rwlock);
|
||||
return QTERROR;
|
||||
}
|
||||
for (i = 0; i < init_userp.thread_nums; i++) {
|
||||
if (qtfs_userps[i].size > QTFS_USERP_MAXSIZE ||
|
||||
!access_ok(qtfs_userps[i].userp, qtfs_userps[i].size) ||
|
||||
!access_ok(qtfs_userps[i].userp2, qtfs_userps[i].size)) {
|
||||
qtfs_err("userp set failed");
|
||||
ret = QTERROR;
|
||||
write_unlock(&g_userp_rwlock);
|
||||
break;
|
||||
}
|
||||
qtfs_info("userp set idx:%d size:%lu", i, qtfs_userps[i].size);
|
||||
}
|
||||
write_unlock(&g_userp_rwlock);
|
||||
break;
|
||||
case QTFS_IOCTL_THREAD_RUN:
|
||||
pvar = qtfs_conn_get_param();
|
||||
if (pvar == NULL)
|
||||
return QTERROR;
|
||||
ret = qtfs_conn_server_run(pvar);
|
||||
if (ret == QTEXIT) {
|
||||
qtfs_warn("qtfs thread idx:%d exit.", pvar->cur_threadidx);
|
||||
qtfs_sm_exit(pvar);
|
||||
qtinfo_cntdec(QTINF_ACTIV_CONN);
|
||||
}
|
||||
qtfs_conn_put_param(pvar);
|
||||
break;
|
||||
case QTFS_IOCTL_EPFDSET:
|
||||
write_lock(&qtfs_epoll_rwlock);
|
||||
if (qtfs_epoll.kevents != NULL) {
|
||||
kfree(qtfs_epoll.kevents);
|
||||
qtfs_epoll.kevents = NULL;
|
||||
}
|
||||
if (copy_from_user(&qtfs_epoll, (void __user *)arg, sizeof(struct qtfs_server_epoll_s))) {
|
||||
qtfs_err("copy epoll struct from arg failed.");
|
||||
ret = QTERROR;
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
break;
|
||||
}
|
||||
if (qtfs_epoll.event_nums > QTFS_MAX_EPEVENTS_NUM || qtfs_epoll.event_nums == 0) {
|
||||
qtfs_err("epoll arg set failed, event nums:%d too big", qtfs_epoll.event_nums);
|
||||
ret = QTERROR;
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
break;
|
||||
}
|
||||
if (qtfs_epoll.epfd < 3) {
|
||||
qtfs_err("epoll epfd set failed, epfd:%d should be greater than 2", qtfs_epoll.epfd);
|
||||
ret = QTERROR;
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
break;
|
||||
}
|
||||
if (!access_ok(qtfs_epoll.events, qtfs_epoll.event_nums * sizeof(struct epoll_event))) {
|
||||
qtfs_err("epoll events set failed, check pointer of qtfs_epoll.events failed");
|
||||
ret = QTERROR;
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
break;
|
||||
}
|
||||
qtfs_info("epoll arg set, epfd:%d event nums:%d events.",
|
||||
qtfs_epoll.epfd, qtfs_epoll.event_nums);
|
||||
qtfs_epoll.kevents = (struct epoll_event *)kmalloc(sizeof(struct epoll_event) *
|
||||
qtfs_epoll.event_nums, GFP_KERNEL);
|
||||
if (qtfs_epoll.kevents == NULL) {
|
||||
qtfs_err("epoll kernel events kmalloc failed.");
|
||||
ret = QTERROR;
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
break;
|
||||
}
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
break;
|
||||
case QTFS_IOCTL_EPOLL_THREAD_INIT:
|
||||
write_lock(&qtfs_epoll_rwlock);
|
||||
ret = qtfs_server_epoll_init();
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
break;
|
||||
case QTFS_IOCTL_EPOLL_THREAD_RUN:
|
||||
write_lock(&qtfs_epoll_rwlock);
|
||||
if (qtfs_epoll_var == NULL) {
|
||||
qtfs_err("qtfs epoll thread run failed, var is invalid.");
|
||||
ret = QTERROR;
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
break;
|
||||
}
|
||||
ret = qtfs_server_epoll_thread(qtfs_epoll_var);
|
||||
write_unlock(&qtfs_epoll_rwlock);
|
||||
if (ret == QTEXIT) {
|
||||
qtfs_info("qtfs epoll thread exit.");
|
||||
qtfs_epoll_cut_conn(qtfs_epoll_var);
|
||||
}
|
||||
break;
|
||||
case QTFS_IOCTL_EXIT:
|
||||
if (arg != 0 && arg != 1) {
|
||||
qtfs_err("qtfs exit input invalid, should be 0 or 1.");
|
||||
ret = QTERROR;
|
||||
break;
|
||||
}
|
||||
qtfs_whitelist_clearall();
|
||||
qtfs_info("qtfs server threads run set to:%lu.", arg);
|
||||
qtfs_server_thread_run = arg;
|
||||
break;
|
||||
case QTFS_IOCTL_ALLINFO:
|
||||
case QTFS_IOCTL_CLEARALL:
|
||||
case QTFS_IOCTL_LOGLEVEL:
|
||||
case QTFS_IOCTL_WL_ADD:
|
||||
case QTFS_IOCTL_WL_DEL:
|
||||
case QTFS_IOCTL_WL_GET:
|
||||
ret = qtfs_misc_ioctl(file, cmd, arg);
|
||||
break;
|
||||
default:
|
||||
qtfs_err("qtfs misc ioctl unknown cmd:%u.", cmd);
|
||||
ret = QTERROR;
|
||||
break;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int __init qtfs_server_init(void)
|
||||
{
|
||||
qtfs_log_init(qtfs_log_level, sizeof(qtfs_log_level));
|
||||
if (qtfs_kallsyms_hack_init() != 0)
|
||||
return -1;
|
||||
rwlock_init(&qtfs_epoll_rwlock);
|
||||
qtfs_whitelist_initset();
|
||||
|
||||
qtfs_diag_info = (struct qtinfo *)kmalloc(sizeof(struct qtinfo), GFP_KERNEL);
|
||||
if (qtfs_diag_info == NULL) {
|
||||
qtfs_err("kmalloc qtfs diag info failed.");
|
||||
return -1;
|
||||
}
|
||||
memset(qtfs_diag_info, 0, sizeof(struct qtinfo));
|
||||
qtfs_userps = (struct qtfs_server_userp_s *)kmalloc(
|
||||
QTFS_MAX_THREADS * sizeof(struct qtfs_server_userp_s), GFP_KERNEL);
|
||||
if (qtfs_userps == NULL) {
|
||||
qtfs_err("kmalloc qtfs userps failed, nums:%d", QTFS_MAX_THREADS);
|
||||
kfree(qtfs_diag_info);
|
||||
return -1;
|
||||
}
|
||||
memset(qtfs_userps, 0, QTFS_MAX_THREADS * sizeof(struct qtfs_server_userp_s));
|
||||
qtfs_fd_bitmap.bitmap = NULL;
|
||||
qtfs_fd_bitmap.nbits = 0;
|
||||
qtfs_conn_param_init();
|
||||
if (qtfs_syscall_replace_start()) {
|
||||
qtfs_err("qtfs syscall replace failed.");
|
||||
goto err_syscall;
|
||||
}
|
||||
if (qtfs_misc_register()) {
|
||||
qtfs_err("qtfs misc device register failed.");
|
||||
goto err_misc;
|
||||
}
|
||||
return 0;
|
||||
err_misc:
|
||||
qtfs_syscall_replace_stop();
|
||||
err_syscall:
|
||||
kfree(qtfs_userps);
|
||||
kfree(qtfs_diag_info);
|
||||
return -1;
|
||||
}
|
||||
|
||||
static void __exit qtfs_server_exit(void)
|
||||
{
|
||||
qtfs_mod_exiting = true;
|
||||
qtfs_server_thread_run = 0;
|
||||
|
||||
qtfs_conn_param_fini();
|
||||
|
||||
if (qtfs_epoll_var != NULL) {
|
||||
qtfs_epoll_cut_conn(qtfs_epoll_var);
|
||||
qtfs_conn_fini(qtfs_epoll_var);
|
||||
qtfs_epoll_var->conn_ops->conn_var_fini(qtfs_epoll_var);
|
||||
kfree(qtfs_epoll_var);
|
||||
qtfs_epoll_var = NULL;
|
||||
}
|
||||
if (qtfs_diag_info != NULL) {
|
||||
kfree(qtfs_diag_info);
|
||||
qtfs_diag_info = NULL;
|
||||
}
|
||||
write_lock(&g_userp_rwlock);
|
||||
if (qtfs_userps != NULL) {
|
||||
kfree(qtfs_userps);
|
||||
qtfs_userps = NULL;
|
||||
}
|
||||
if (qtfs_fd_bitmap.bitmap != NULL) {
|
||||
kfree(qtfs_fd_bitmap.bitmap);
|
||||
qtfs_fd_bitmap.bitmap = NULL;
|
||||
qtfs_fd_bitmap.nbits = 0;
|
||||
}
|
||||
write_unlock(&g_userp_rwlock);
|
||||
qtfs_whitelist_exit();
|
||||
qtfs_server_epoll_exit();
|
||||
|
||||
qtfs_misc_destroy();
|
||||
qtfs_syscall_replace_stop();
|
||||
qtfs_info("qtfs server exit done.\n");
|
||||
return;
|
||||
}
|
||||
|
||||
module_init(qtfs_server_init);
|
||||
module_exit(qtfs_server_exit);
|
||||
MODULE_AUTHOR("liqiang64@huawei.com");
|
||||
MODULE_LICENSE("GPL");
|
51
qtfs/qtfs_server/qtfs-server.h
Normal file
51
qtfs/qtfs_server/qtfs-server.h
Normal file
@ -0,0 +1,51 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#ifndef __QTFS_SERVER_H__
|
||||
#define __QTFS_SERVER_H__
|
||||
|
||||
extern int qtfs_server_thread_run;
|
||||
extern struct qtfs_server_epoll_s qtfs_epoll;
|
||||
extern int qtfs_mod_exiting;
|
||||
extern rwlock_t g_userp_rwlock;
|
||||
extern struct qtserver_fd_bitmap qtfs_fd_bitmap;
|
||||
|
||||
struct qtserver_arg {
|
||||
char *data;
|
||||
char *out;
|
||||
struct qtfs_server_userp_s *userp;
|
||||
};
|
||||
|
||||
struct qtserver_ops {
|
||||
int type;
|
||||
int (*precheck) (void *); // check req from socket recv
|
||||
// return int is output len.
|
||||
int (*handle) (struct qtserver_arg *);
|
||||
char str[32];
|
||||
};
|
||||
|
||||
struct qtserver_fd_bitmap {
|
||||
unsigned int nbits;
|
||||
unsigned long *bitmap;
|
||||
};
|
||||
|
||||
int qtfs_conn_server_run(struct qtfs_conn_var_s *pvar);
|
||||
long qtfs_misc_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
int qtfs_misc_register(void);
|
||||
void qtfs_misc_destroy(void);
|
||||
void qtfs_whitelist_exit(void);
|
||||
long qtfs_misc_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
void qtfs_whitelist_clearall(void);
|
||||
void qtfs_whitelist_initset(void);
|
||||
|
||||
#endif
|
9
qtfs/qtfs_server/qtfs_fifo_server/Cargo.toml
Normal file
9
qtfs/qtfs_server/qtfs_fifo_server/Cargo.toml
Normal file
@ -0,0 +1,9 @@
|
||||
[package]
|
||||
name = "qtfs_fifo_server"
|
||||
version = "1.0.0"
|
||||
edition = "2021"
|
||||
|
||||
[dependencies]
|
||||
tokio = { version = "1.29.1", features = ["full"]}
|
||||
libc = "0.2"
|
||||
rlimit = "0.10.1"
|
337
qtfs/qtfs_server/qtfs_fifo_server/src/cofifo.rs
Normal file
337
qtfs/qtfs_server/qtfs_fifo_server/src/cofifo.rs
Normal file
@ -0,0 +1,337 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-07-26
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
use tokio::net::TcpStream;
|
||||
use std::mem;
|
||||
use tokio::fs::File;
|
||||
use tokio::fs;
|
||||
use std::os::unix::fs::FileTypeExt;
|
||||
use tokio::fs::OpenOptions;
|
||||
|
||||
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
||||
|
||||
#[derive(Debug, Clone, Copy)]
|
||||
#[repr(C, packed)]
|
||||
struct Qtreq {
|
||||
// magic: [u8; 4], //magic: 0x5aa55aa5
|
||||
msgtype: u32,
|
||||
error: u32,
|
||||
seq_num: u64,
|
||||
len: usize,
|
||||
}
|
||||
|
||||
const QTFS_REQ_OPEN: u32 = 2;
|
||||
const QTFS_REQ_CLOSE: u32 = 3;
|
||||
const QTFS_REQ_READ: u32 = 5;
|
||||
const QTFS_REQ_WRITE: u32 = 6;
|
||||
pub async fn qtfs_fifo_server(stream: TcpStream, idx: usize) {
|
||||
let mut conn = Conn {stream};
|
||||
let mut head: Qtreq;
|
||||
|
||||
match conn.qtfs_req_head(idx.clone()).await {
|
||||
Ok(h) => head = h,
|
||||
Err(e) => {
|
||||
println!("Recv invalid head exit this proc :{}.", e);
|
||||
return;
|
||||
}
|
||||
}
|
||||
if head.msgtype != QTFS_REQ_OPEN {
|
||||
println!("first msg type is invalid");
|
||||
return;
|
||||
}
|
||||
let file = match conn.qtfs_fifo_open(head.clone()).await {
|
||||
Ok(f) => {
|
||||
head.len = mem::size_of::<Qtrspopen>();
|
||||
conn.req_head_ack(head).await;
|
||||
conn.open_ack(0).await;
|
||||
f
|
||||
}
|
||||
Err(e) => {
|
||||
head.len = mem::size_of::<Qtrspopen>();
|
||||
println!("Open fifo error:{}", e);
|
||||
conn.req_head_ack(head).await;
|
||||
conn.open_ack(1).await;
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
'main: loop {
|
||||
let mut head: Qtreq;
|
||||
match conn.qtfs_req_head(idx.clone()).await {
|
||||
Ok(h) => head = h,
|
||||
Err(e) => {
|
||||
println!("head recv failed, {}.", e);
|
||||
return;
|
||||
}
|
||||
}
|
||||
match head.msgtype {
|
||||
QTFS_REQ_OPEN => {
|
||||
println!("Fifo is opened and recv open request again!");
|
||||
head.len = mem::size_of::<Qtrspopen>();
|
||||
conn.req_head_ack(head).await;
|
||||
conn.open_ack(1).await;
|
||||
}
|
||||
QTFS_REQ_CLOSE => {
|
||||
println!("Close req idx:{}", idx.clone());
|
||||
head.len = 0;
|
||||
conn.req_head_ack(head).await;
|
||||
break 'main;
|
||||
}
|
||||
QTFS_REQ_READ => {
|
||||
println!("Read req idx:{}", idx.clone());
|
||||
conn.qtfs_fifo_read(file.try_clone().await.unwrap(), head).await;
|
||||
}
|
||||
QTFS_REQ_WRITE => {
|
||||
println!("Write req idx:{}", idx.clone());
|
||||
conn.qtfs_fifo_write(file.try_clone().await.unwrap(), head).await;
|
||||
}
|
||||
_ => {
|
||||
println!("Recv invalid msg type");
|
||||
}
|
||||
}
|
||||
}
|
||||
println!("Fifo server idx:{} is closed.", idx);
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Copy)]
|
||||
#[repr(C, packed)]
|
||||
struct Qtreqopen {
|
||||
flags: u64,
|
||||
mode: u32,
|
||||
}
|
||||
#[repr(C, packed)]
|
||||
struct Qtrspopen {
|
||||
ret: i32,
|
||||
}
|
||||
|
||||
#[repr(C, packed)]
|
||||
struct Qtreqread {
|
||||
len: u64,
|
||||
}
|
||||
|
||||
#[repr(C, packed)]
|
||||
struct Qtrspread {
|
||||
errno: i32,
|
||||
len: u64,
|
||||
}
|
||||
|
||||
|
||||
|
||||
#[repr(C, packed)]
|
||||
struct Qtreqwrite {
|
||||
len: u64,
|
||||
}
|
||||
#[repr(C, packed)]
|
||||
struct Qtrspwrite {
|
||||
errno: i32,
|
||||
len: u64,
|
||||
}
|
||||
|
||||
struct Conn {
|
||||
stream: TcpStream,
|
||||
}
|
||||
|
||||
impl Conn {
|
||||
// sync head magic bytes sequence: 0x5a 0xa5 0x5a 0xa5
|
||||
// 逐字节读取magic,连续匹配的四个字节即视为同步包头
|
||||
async fn package_sync(&mut self) {
|
||||
let mut byte: [u8; 1] = [0; 1];
|
||||
loop {
|
||||
self.stream.read_exact(&mut byte).await.unwrap();
|
||||
if byte[0] != 0x5a {continue;}
|
||||
self.stream.read_exact(&mut byte).await.unwrap();
|
||||
if byte[0] != 0xa5 {continue;}
|
||||
self.stream.read_exact(&mut byte).await.unwrap();
|
||||
if byte[0] != 0x5a {continue;}
|
||||
self.stream.read_exact(&mut byte).await.unwrap();
|
||||
if byte[0] != 0xa5 {continue;}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
async fn send_magic_head(&mut self) {
|
||||
const MAGIC: [u8; 4] = [0x5a, 0xa5, 0x5a, 0xa5];
|
||||
let _ = self.stream.write_all(&MAGIC[0..4]).await;
|
||||
}
|
||||
|
||||
async fn qtfs_req_head(&mut self, _idx: usize) -> Result<Qtreq, std::io::Error> {
|
||||
const HEADSIZE: usize = mem::size_of::<Qtreq>();
|
||||
self.package_sync().await;
|
||||
let mut msghead = [0; HEADSIZE];
|
||||
self.stream.read_exact(&mut msghead).await?;
|
||||
let head = Qtreq {
|
||||
msgtype: u32::from_le_bytes(msghead[0..4].try_into().unwrap()),
|
||||
error: u32::from_le_bytes(msghead[4..8].try_into().unwrap()),
|
||||
seq_num: u64::from_le_bytes(msghead[8..16].try_into().unwrap()),
|
||||
len: usize::from_le_bytes(msghead[16..16+mem::size_of::<usize>()].try_into().unwrap()),
|
||||
};
|
||||
let reqtype: String = match head.msgtype {
|
||||
QTFS_REQ_OPEN => String::from("Open"),
|
||||
QTFS_REQ_CLOSE => String::from("Close"),
|
||||
QTFS_REQ_READ => String::from("Read"),
|
||||
QTFS_REQ_WRITE => String::from("Write"),
|
||||
_ => String::from("Unknown"),
|
||||
};
|
||||
println!("Recv new head type:{} msg:{:?}", reqtype, head);
|
||||
Ok(head)
|
||||
}
|
||||
|
||||
async fn qtfs_fifo_open(&mut self, head: Qtreq) -> Result<File, i32> {
|
||||
const HEADSIZE: usize = mem::size_of::<Qtreqopen>();
|
||||
let mut openhead = [0; HEADSIZE];
|
||||
|
||||
if head.len >= 4096 + HEADSIZE {
|
||||
println!("qtfs fifo len invalid");
|
||||
return Err(1);
|
||||
}
|
||||
self.stream.read_exact(&mut openhead).await.unwrap();
|
||||
let openhead1 = Qtreqopen {
|
||||
flags: u64::from_le_bytes(openhead[0..8].try_into().unwrap()),
|
||||
mode: u32::from_le_bytes(openhead[8..12].try_into().unwrap()),
|
||||
};
|
||||
println!("open head:{:?}", openhead1);
|
||||
let mut path = Vec::with_capacity(head.len - HEADSIZE);
|
||||
path.resize(head.len - HEADSIZE, 0);
|
||||
self.stream.read_exact(&mut path).await.unwrap();
|
||||
|
||||
let getstr = String::from_utf8(path).unwrap();
|
||||
let pathstr = getstr.trim_end_matches('\0').trim();
|
||||
match fs::metadata(pathstr.clone()).await {
|
||||
Ok(meta) => {
|
||||
if meta.file_type().is_fifo() == false {
|
||||
println!("Requst path:{} not fifo!", pathstr);
|
||||
return Err(1);
|
||||
}
|
||||
}
|
||||
Err(_) => {
|
||||
println!("path:{} check failed.", pathstr);
|
||||
return Err(1);
|
||||
}
|
||||
};
|
||||
println!("Recv open path:{}", pathstr);
|
||||
let file = OpenOptions::new()
|
||||
.read(true)
|
||||
.write(true)
|
||||
//.custom_flags(libc::O_NONBLOCK)
|
||||
.open(pathstr).await.unwrap();
|
||||
|
||||
Ok(file)
|
||||
}
|
||||
|
||||
async fn qtfs_fifo_read(&mut self, mut file: File, mut reqhead: Qtreq) {
|
||||
let mut head = [0; mem::size_of::<Qtreqread>()];
|
||||
self.stream.read_exact(&mut head).await.unwrap();
|
||||
let req = Qtreqread {
|
||||
len: u64::from_le_bytes(head[0..8].try_into().unwrap()),
|
||||
};
|
||||
let len = std::cmp::min(req.len, 4096);
|
||||
|
||||
let mut rsp = Qtrspread {
|
||||
errno: 0,
|
||||
len: 0,
|
||||
};
|
||||
|
||||
let mut buf = Vec::with_capacity(len.try_into().unwrap());
|
||||
buf.resize(len.try_into().unwrap(), 0);
|
||||
|
||||
match file.read(&mut buf).await {
|
||||
Ok(n) => {
|
||||
rsp.len = n as u64;
|
||||
let send = unsafe {
|
||||
let ptr = &rsp as *const Qtrspread as *const u8;
|
||||
std::slice::from_raw_parts(ptr, mem::size_of::<Qtrspread>())
|
||||
};
|
||||
println!("Read {} bytes from fifo", n.clone());
|
||||
reqhead.len = mem::size_of::<Qtrspread>();
|
||||
self.req_head_ack(reqhead).await;
|
||||
self.stream.write_all(&send[..mem::size_of::<Qtrspread>()]).await.unwrap();
|
||||
let _ = self.stream.write_all(&buf[..n]).await.unwrap();
|
||||
}
|
||||
Err(e) => {
|
||||
rsp.errno = -1;
|
||||
rsp.len = 0;
|
||||
let send = unsafe {
|
||||
let ptr = &rsp as *const Qtrspread as *const u8;
|
||||
std::slice::from_raw_parts(ptr, mem::size_of::<Qtrspread>())
|
||||
};
|
||||
reqhead.len = mem::size_of::<Qtrspread>();
|
||||
self.req_head_ack(reqhead).await;
|
||||
self.stream.write_all(&send[..mem::size_of::<Qtrspread>()]).await.unwrap();
|
||||
println!("Read from fifo error:{}", e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
async fn qtfs_fifo_write(&mut self, mut file: File, mut reqhead: Qtreq) {
|
||||
let mut whead = [0; mem::size_of::<Qtreqwrite>()];
|
||||
self.stream.read_exact(&mut whead).await.unwrap();
|
||||
let len = u64::from_le_bytes(whead[0..8].try_into().unwrap());
|
||||
|
||||
// 最大接收一次性写入4k
|
||||
let len = std::cmp::min(len, 4096);
|
||||
|
||||
let mut rsp = Qtrspwrite {
|
||||
errno: 0,
|
||||
len: 0,
|
||||
};
|
||||
|
||||
let mut buf = Vec::with_capacity(len.try_into().unwrap());
|
||||
buf.resize(len.try_into().unwrap(), 0);
|
||||
self.stream.read_exact(&mut buf).await.unwrap();
|
||||
|
||||
match file.write_all(&mut buf[..len as usize]).await {
|
||||
Ok(_) => {
|
||||
rsp.len = len as u64;
|
||||
reqhead.len = mem::size_of::<Qtrspwrite>();
|
||||
self.req_head_ack(reqhead).await;
|
||||
self.write_ack(rsp).await;
|
||||
println!("Write fifo ok, send ack.");
|
||||
}
|
||||
Err(e) => {
|
||||
rsp.len = 0;
|
||||
reqhead.len = mem::size_of::<Qtrspwrite>();
|
||||
self.req_head_ack(reqhead).await;
|
||||
self.write_ack(rsp).await;
|
||||
println!("Write failed {}.", e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
async fn req_head_ack(&mut self, head: Qtreq) {
|
||||
let send = unsafe {
|
||||
let ptr = &head as *const Qtreq as *const u8;
|
||||
std::slice::from_raw_parts(ptr, mem::size_of::<Qtreq>())
|
||||
};
|
||||
self.send_magic_head().await;
|
||||
self.stream.write_all(&send[..mem::size_of::<Qtreq>()]).await.expect("req head ack failed");
|
||||
}
|
||||
|
||||
async fn open_ack(&mut self, retcode: i32) {
|
||||
let rsp = Qtrspopen {ret: retcode,};
|
||||
let send = unsafe {
|
||||
let ptr = &rsp as *const Qtrspopen as *const u8;
|
||||
std::slice::from_raw_parts(ptr, mem::size_of::<Qtrspopen>())
|
||||
};
|
||||
self.stream.write_all(&send[..mem::size_of::<Qtrspopen>()]).await.expect("Response open failed");
|
||||
}
|
||||
|
||||
async fn write_ack(&mut self, rsp: Qtrspwrite) {
|
||||
let send = unsafe {
|
||||
let ptr = &rsp as *const Qtrspwrite as *const u8;
|
||||
std::slice::from_raw_parts(ptr, mem::size_of::<Qtrspwrite>())
|
||||
};
|
||||
self.stream.write_all(&send[..mem::size_of::<Qtrspwrite>()]).await.expect("Response write failed");
|
||||
}
|
||||
}
|
72
qtfs/qtfs_server/qtfs_fifo_server/src/main.rs
Normal file
72
qtfs/qtfs_server/qtfs_fifo_server/src/main.rs
Normal file
@ -0,0 +1,72 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-07-26
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
use std::env;
|
||||
use std::net::TcpListener;
|
||||
use tokio::net::TcpListener as AsyncTcpListener;
|
||||
use tokio::runtime::Builder;
|
||||
|
||||
extern crate rlimit;
|
||||
use rlimit::Resource;
|
||||
use rlimit::setrlimit;
|
||||
mod cofifo;
|
||||
|
||||
async fn set_rlimit_fd(){
|
||||
let rlimit = Resource::NOFILE;
|
||||
let fd_limit = 65536;
|
||||
match setrlimit(rlimit, fd_limit, fd_limit) {
|
||||
Ok(_) => {},
|
||||
Err(e) => println!("Set file rlimit to {} failed {}.", fd_limit, e),
|
||||
}
|
||||
}
|
||||
|
||||
#[tokio::main]
|
||||
async fn main() {
|
||||
let args: Vec<String> = env::args().collect();
|
||||
if args.len() != 3 {
|
||||
let bin: String = args[0].trim().parse().expect("Binary name error");
|
||||
println!("Usage example:");
|
||||
println!(" {} 192.168.1.10:12310 10", bin);
|
||||
return;
|
||||
}
|
||||
set_rlimit_fd().await;
|
||||
let addr: String = args[1].trim().parse().expect("Input address: '192.168.1.10:12310'");
|
||||
let max_block_threads: usize = args[2].trim().parse().expect("Input max blocking threads number in arg 2: like '10'");
|
||||
let listener = TcpListener::bind(addr.clone()).unwrap();
|
||||
let async_listener = AsyncTcpListener::from_std(listener).unwrap();
|
||||
let runtime = Builder::new_multi_thread()
|
||||
.max_blocking_threads(max_block_threads)
|
||||
.enable_all()
|
||||
.build()
|
||||
.unwrap();
|
||||
|
||||
println!("Ready to listen addr:{}, max blocking threads:{}", addr, max_block_threads);
|
||||
|
||||
let mut coroutine_idx: usize = 1;
|
||||
loop {
|
||||
let (s, _) = async_listener.accept().await.unwrap();
|
||||
let cur_idx = coroutine_idx.clone();
|
||||
coroutine_idx += 1;
|
||||
match Some(s) {
|
||||
Some(stream) => {
|
||||
// 收到一个新的fifo连接请求,拉起新的协程处理函数
|
||||
runtime.spawn(cofifo::qtfs_fifo_server(stream, cur_idx));
|
||||
}
|
||||
_ => {
|
||||
eprintln!("Accept error!");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
23
qtfs/qtinfo/Makefile
Normal file
23
qtfs/qtinfo/Makefile
Normal file
@ -0,0 +1,23 @@
|
||||
CFLAGS += -g -O2
|
||||
CFLAGS += -fstack-protector-strong
|
||||
CFLAGS += -fPIE -pie -fPIC
|
||||
CFLAGS += -D_FORTIFY_SOURCE=2
|
||||
LDFLAGS += -Wl,-z,now
|
||||
LDFLAGS += -Wl,-z,noexecstack
|
||||
LDFLAGS += -fPIE -pie
|
||||
|
||||
all: qtinfo qtcfg
|
||||
|
||||
qtinfo:
|
||||
gcc $(CFLAGS) $(LDFLAGS) -D$(role) -o qtinfo qtinfo.c -I../ -I../include/
|
||||
|
||||
qtcfg:
|
||||
gcc $(CFLAGS) $(LDFLAGS) -DQTINFO_RELEASE -D$(role) -o qtcfg qtinfo.c -I../ -I../include/
|
||||
|
||||
install:
|
||||
yes | cp qtinfo /usr/bin/
|
||||
yes | cp qtcfg /usr/bin/
|
||||
|
||||
clean:
|
||||
rm -rf qtcfg qtinfo qtinfo.o
|
||||
|
627
qtfs/qtinfo/qtinfo.c
Normal file
627
qtfs/qtinfo/qtinfo.c
Normal file
@ -0,0 +1,627 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <string.h>
|
||||
#include <stddef.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/un.h>
|
||||
|
||||
#include "qtinfo.h"
|
||||
#include "comm.h"
|
||||
#include "ipc/uds_module.h"
|
||||
|
||||
#ifdef client
|
||||
#define QTFS_DEV_NAME "/dev/qtfs_client"
|
||||
#else
|
||||
#define QTFS_DEV_NAME "/dev/qtfs_server"
|
||||
#endif
|
||||
|
||||
#define qtinfo_out(info, ...) \
|
||||
do {\
|
||||
printf(info"\n", ##__VA_ARGS__);\
|
||||
} while (0);
|
||||
|
||||
#define qtinfo_out2(info, ...) \
|
||||
do {\
|
||||
printf(info, ##__VA_ARGS__);\
|
||||
} while (0);
|
||||
|
||||
#define qtinfo_err(info, ...) \
|
||||
do {\
|
||||
printf("ERROR: "info"\n", ##__VA_ARGS__);\
|
||||
} while (0);
|
||||
|
||||
#define RECV_BUFF_LEN 256
|
||||
|
||||
struct qtinfo_type_str qtinfo_all_events[] = {
|
||||
{QTFS_REQ_NULL, "null"},
|
||||
{QTFS_REQ_MOUNT, "mount"},
|
||||
{QTFS_REQ_OPEN, "open"},
|
||||
{QTFS_REQ_CLOSE, "close"},
|
||||
{QTFS_REQ_READ, "read"},
|
||||
{QTFS_REQ_READITER, "readiter"}, //5
|
||||
{QTFS_REQ_WRITE, "write"},
|
||||
{QTFS_REQ_LOOKUP, "lookup"},
|
||||
{QTFS_REQ_READDIR, "readdir"},
|
||||
{QTFS_REQ_MKDIR, "mkdir"},
|
||||
{QTFS_REQ_RMDIR, "rmdir"}, //10
|
||||
{QTFS_REQ_GETATTR, "getattr"},
|
||||
{QTFS_REQ_SETATTR, "setattr"},
|
||||
{QTFS_REQ_ICREATE, "icreate"},
|
||||
{QTFS_REQ_MKNOD, "mknod"},
|
||||
{QTFS_REQ_UNLINK, "unlink"}, //15
|
||||
{QTFS_REQ_SYMLINK, "symlink"},
|
||||
{QTFS_REQ_LINK, "link"},
|
||||
{QTFS_REQ_GETLINK, "getlink"},
|
||||
{QTFS_REQ_READLINK, "readlink"},
|
||||
{QTFS_REQ_RENAME, "rename"}, //20
|
||||
|
||||
{QTFS_REQ_XATTRLIST, "xattrlist"},
|
||||
{QTFS_REQ_XATTRGET, "xattrget"},
|
||||
{QTFS_REQ_XATTRSET, "xattrset"},
|
||||
|
||||
{QTFS_REQ_SYSMOUNT, "sysmount"},
|
||||
{QTFS_REQ_SYSUMOUNT, "sysumount"}, //25
|
||||
{QTFS_REQ_FIFOPOLL, "fifo_poll"},
|
||||
|
||||
{QTFS_REQ_STATFS, "statfs"},
|
||||
{QTFS_REQ_IOCTL, "ioctl"},
|
||||
{QTFS_REQ_EPOLL_CTL, "epollctl"},
|
||||
|
||||
{QTFS_REQ_EPOLL_EVENT, "epollevent"}, // 30
|
||||
{QTFS_REQ_LLSEEK, "llseek"},
|
||||
{QTFS_SC_KILL, "sc_kill"},
|
||||
{QTFS_SC_SCHED_GETAFFINITY, "sc_getaffi"},
|
||||
{QTFS_SC_SCHED_SETAFFINITY, "sc_setaffi"},
|
||||
};
|
||||
|
||||
static void qtinfo_events_count(struct qtinfo *evts)
|
||||
{
|
||||
unsigned long total = 0;
|
||||
int i;
|
||||
#ifdef client
|
||||
qtinfo_out("++++++++++++++++++++++++++send events count++++++++++++++++++++++++++");
|
||||
for (i = 0; i < (sizeof(qtinfo_all_events) / sizeof(struct qtinfo_type_str)) - 3; i += 3) {
|
||||
qtinfo_out("%-10s: %-10lu %-10s: %-10lu %-10s: %-10lu",
|
||||
qtinfo_all_events[i].str, evts->c.o_events[i],
|
||||
qtinfo_all_events[i+1].str, evts->c.o_events[i+1],
|
||||
qtinfo_all_events[i+2].str, evts->c.o_events[i+2]);
|
||||
total += evts->c.o_events[i] + evts->c.o_events[i+1] + evts->c.o_events[i+2];
|
||||
}
|
||||
for (; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s: %-10lu ", qtinfo_all_events[i].str, evts->c.o_events[i]);
|
||||
total += evts->c.o_events[i];
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
qtinfo_out("Send events total: %lu", total);
|
||||
total = 0;
|
||||
|
||||
qtinfo_out("++++++++++++++++++++++++++recv events count++++++++++++++++++++++++++");
|
||||
for (i = 0; i < (sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str)) - 3; i+=3) {
|
||||
qtinfo_out("%-10s: %-10lu %-10s: %-10lu %-10s: %-10lu",
|
||||
qtinfo_all_events[i].str, evts->c.i_events[i],
|
||||
qtinfo_all_events[i+1].str, evts->c.i_events[i+1],
|
||||
qtinfo_all_events[i+2].str, evts->c.i_events[i+2]);
|
||||
total += evts->c.i_events[i] + evts->c.i_events[i+1] + evts->c.i_events[i+2];
|
||||
}
|
||||
for(; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s: %-10lu ", qtinfo_all_events[i].str, evts->c.i_events[i]);
|
||||
total += evts->c.i_events[i];
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
qtinfo_out("Recv events total: %lu", total);
|
||||
total = 0;
|
||||
qtinfo_out("++++++++++++++++++++++++++send error count+++++++++++++++++++++++++++");
|
||||
for(i = 0; i < (sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str)) - 3; i+=3) {
|
||||
qtinfo_out("%-10s: %-10lu %-10s: %-10lu %-10s: %-10lu",
|
||||
qtinfo_all_events[i].str, evts->c.send_err[i],
|
||||
qtinfo_all_events[i+1].str, evts->c.send_err[i+1],
|
||||
qtinfo_all_events[i+2].str, evts->c.send_err[i+2]);
|
||||
total += evts->c.send_err[i] + evts->c.send_err[i+1] + evts->c.send_err[i+2];
|
||||
}
|
||||
for(; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s: %-10lu ", qtinfo_all_events[i].str, evts->c.send_err[i]);
|
||||
total += evts->c.send_err[i];
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
qtinfo_out("Send events error total: %lu", total);
|
||||
total = 0;
|
||||
|
||||
qtinfo_out("++++++++++++++++++++++++++recv error count+++++++++++++++++++++++++++");
|
||||
for(i = 0; i < (sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str)) - 3; i+=3) {
|
||||
qtinfo_out("%-10s: %-10lu %-10s: %-10lu %-10s: %-10lu",
|
||||
qtinfo_all_events[i].str, evts->c.recv_err[i],
|
||||
qtinfo_all_events[i+1].str, evts->c.recv_err[i+1],
|
||||
qtinfo_all_events[i+2].str, evts->c.recv_err[i+2]);
|
||||
total += evts->c.recv_err[i] + evts->c.recv_err[i+1] + evts->c.recv_err[i+2];
|
||||
}
|
||||
for(; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s: %-10lu ", qtinfo_all_events[i].str, evts->c.recv_err[i]);
|
||||
total += evts->c.recv_err[i];
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
qtinfo_out("Recv events error total: %lu", total);
|
||||
total = 0;
|
||||
#endif
|
||||
#ifdef server
|
||||
qtinfo_out("++++++++++++++++++++++++++send events count++++++++++++++++++++++++++");
|
||||
for (i = 0; i < (sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str)) - 3; i+=3) {
|
||||
qtinfo_out("%-10s: %-10lu %-10s: %-10lu %-10s: %-10lu",
|
||||
qtinfo_all_events[i].str, evts->s.o_events[i],
|
||||
qtinfo_all_events[i+1].str, evts->s.o_events[i+1],
|
||||
qtinfo_all_events[i+2].str, evts->s.o_events[i+2]);
|
||||
total += evts->s.o_events[i] + evts->s.o_events[i+1] + evts->s.o_events[i+2];
|
||||
}
|
||||
for (; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s: %-10lu ", qtinfo_all_events[i].str, evts->s.o_events[i]);
|
||||
total += evts->s.o_events[i];
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
qtinfo_out("Send events total: %lu", total);
|
||||
total = 0;
|
||||
|
||||
qtinfo_out("++++++++++++++++++++++++++recv events count++++++++++++++++++++++++++");
|
||||
for (i = 0; i < (sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str)) - 3; i+=3) {
|
||||
qtinfo_out("%-10s: %-10lu %-10s: %-10lu %-10s: %-10lu",
|
||||
qtinfo_all_events[i].str, evts->s.i_events[i],
|
||||
qtinfo_all_events[i+1].str, evts->s.i_events[i+1],
|
||||
qtinfo_all_events[i+2].str, evts->s.i_events[i+2]);
|
||||
total += evts->s.i_events[i] + evts->s.i_events[i+1] + evts->s.i_events[i+2];
|
||||
}
|
||||
for(; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s: %-10lu ", qtinfo_all_events[i].str, evts->s.i_events[i]);
|
||||
total += evts->s.i_events[i];
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
qtinfo_out("Recv events total: %lu", total);
|
||||
|
||||
qtinfo_out("++++++++++++++++++++++++++req check err+++++++++++++++++++++++++++++++");
|
||||
for (i = 0; i < (sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str)) - 3; i+=3) {
|
||||
qtinfo_out("%-10s: %-10lu %-10s: %-10lu %-10s: %-10lu",
|
||||
qtinfo_all_events[i].str, evts->s.req_check[i],
|
||||
qtinfo_all_events[i+1].str, evts->s.req_check[i+1],
|
||||
qtinfo_all_events[i+2].str, evts->s.req_check[i+2]);
|
||||
}
|
||||
for(; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s: %-10lu ", qtinfo_all_events[i].str, evts->s.req_check[i]);
|
||||
}
|
||||
qtinfo_out("\n+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++")
|
||||
#endif
|
||||
}
|
||||
|
||||
static void qtinfo_misc_count(struct qtinfo *info)
|
||||
{
|
||||
qtinfo_out("++++++++++++++++++++++++++++++Misc count+++++++++++++++++++++++++++++");
|
||||
#ifdef client
|
||||
qtinfo_out("Active connects: %-8lu Seq err count : %-8lu Restartsys : %-8lu",
|
||||
info->c.cnts[QTINF_ACTIV_CONN], info->c.cnts[QTINF_SEQ_ERR], info->c.cnts[QTINF_RESTART_SYS]);
|
||||
qtinfo_out("Type mismatch : %-8lu Epoll add fds : %-8lu Epoll del fds: %-8lu",
|
||||
info->c.cnts[QTINF_TYPE_MISMATCH], info->c.cnts[QTINF_EPOLL_ADDFDS], info->c.cnts[QTINF_EPOLL_DELFDS]);
|
||||
qtinfo_out("Epoll err fds : %-8lu",
|
||||
info->c.cnts[QTINF_EPOLL_FDERR]);
|
||||
#else
|
||||
qtinfo_out("Active connects: %-8lu Epoll add fds: %-8lu Epoll del fds: %-8lu",
|
||||
info->s.cnts[QTINF_ACTIV_CONN], info->s.cnts[QTINF_EPOLL_ADDFDS], info->s.cnts[QTINF_EPOLL_DELFDS]);
|
||||
#endif
|
||||
}
|
||||
|
||||
static void qtinfo_thread_state(struct qtinfo *info)
|
||||
{
|
||||
int i = 0;
|
||||
|
||||
qtinfo_out("+++++++++++++++++++++++++++Connection state++++++++++++++++++++++++++");
|
||||
for (i = 0; i < QTFS_MAX_THREADS - 3; i+=3) {
|
||||
qtinfo_out("Conn%-2d: %-10s Conn%-2d: %-10s Conn%-2d: %-10s",
|
||||
i+1, QTINFO_STATE(info->thread_state[i]),
|
||||
i+2, QTINFO_STATE(info->thread_state[i+1]),
|
||||
i+3, QTINFO_STATE(info->thread_state[i+2]));
|
||||
}
|
||||
for (; i < QTFS_MAX_THREADS; i++) {
|
||||
qtinfo_out2("Conn%-2d: %-10s ", i+1, QTINFO_STATE(info->thread_state[i]));
|
||||
}
|
||||
qtinfo_out("Epoll state: %-10s", QTINFO_STATE(info->epoll_state));
|
||||
return;
|
||||
}
|
||||
|
||||
static void qtinfo_pvar_count(struct qtinfo *info)
|
||||
{
|
||||
int i = 0;
|
||||
qtinfo_out("+++++++++++++++++++++++++++++Param count+++++++++++++++++++++++++++++");
|
||||
qtinfo_out("Parameter valid count: %-2d Parameter busy count: %-2d",
|
||||
info->pvar_vld, info->pvar_busy);
|
||||
for (i = 0; i < QTFS_MAX_THREADS; i++) {
|
||||
qtinfo_out("Conn%-2d holder: [%-20s]", i+1, (info->who_using[i][0] == '\0') ? "No one" : info->who_using[i]);
|
||||
}
|
||||
}
|
||||
|
||||
static void qtinfo_log_level(struct qtinfo *info)
|
||||
{
|
||||
qtinfo_out("Log level: %d", info->log_level);
|
||||
}
|
||||
|
||||
static int qtinfo_opt_a(int fd)
|
||||
{
|
||||
struct qtinfo *diag = (struct qtinfo *)malloc(sizeof(struct qtinfo));
|
||||
if (diag == NULL) {
|
||||
qtinfo_err("malloc failed.");
|
||||
return -1;
|
||||
}
|
||||
memset(diag, 0, sizeof(struct qtinfo));
|
||||
int ret = ioctl(fd, QTFS_IOCTL_ALLINFO, diag);
|
||||
if (ret != QTOK) {
|
||||
qtinfo_err("ioctl failed, ret:%d.", ret);
|
||||
free(diag);
|
||||
return -1;
|
||||
}
|
||||
qtinfo_events_count(diag);
|
||||
qtinfo_misc_count(diag);
|
||||
qtinfo_log_level(diag);
|
||||
qtinfo_thread_state(diag);
|
||||
qtinfo_pvar_count(diag);
|
||||
free(diag);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int qtinfo_opt_c(int fd)
|
||||
{
|
||||
int ret = ioctl(fd, QTFS_IOCTL_CLEARALL, NULL);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int qtinfo_opt_l(int fd, char *level)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = ioctl(fd, QTFS_IOCTL_LOGLEVEL, level);
|
||||
if (ret != QTOK) {
|
||||
qtinfo_err("Set qtfs log level:%s failed.", level);
|
||||
return ret;
|
||||
}
|
||||
qtinfo_out("Set qtfs log level to %s success.", level);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int qtinfo_opt_t(int fd)
|
||||
{
|
||||
int i;
|
||||
struct qtinfo *diag = (struct qtinfo *)malloc(sizeof(struct qtinfo));
|
||||
if (diag == NULL) {
|
||||
qtinfo_err("malloc failed.");
|
||||
return -1;
|
||||
}
|
||||
int ret = ioctl(fd, QTFS_IOCTL_ALLINFO, (unsigned long)diag);
|
||||
qtinfo_out("++++++++++++++++++++++++++qtreq_xxx size++++++++++++++++++++++++++");
|
||||
for (i = 0; i < (sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str)) - 3; i+=3) {
|
||||
qtinfo_out("%-10s req: %-10lu %-10s req: %-10lu %-10s req: %-10lu",
|
||||
qtinfo_all_events[i].str, diag->req_size[i],
|
||||
qtinfo_all_events[i+1].str, diag->req_size[i+1],
|
||||
qtinfo_all_events[i+2].str, diag->req_size[i+2]);
|
||||
}
|
||||
for (; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s req: %-10lu ", qtinfo_all_events[i].str, diag->req_size[i]);
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
qtinfo_out("++++++++++++++++++++++++++qtrsp_xxx size++++++++++++++++++++++++++");
|
||||
for (i = 0; i < (sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str)) - 3; i+=3) {
|
||||
qtinfo_out("%-10s rsp: %-10lu %-10s rsp: %-10lu %-10s rsp: %-10lu",
|
||||
qtinfo_all_events[i].str, diag->rsp_size[i],
|
||||
qtinfo_all_events[i+1].str, diag->rsp_size[i+1],
|
||||
qtinfo_all_events[i+2].str, diag->rsp_size[i+2]);
|
||||
}
|
||||
for (; i < sizeof(qtinfo_all_events)/sizeof(struct qtinfo_type_str); i++) {
|
||||
qtinfo_out2("%-10s rsp: %-10lu ", qtinfo_all_events[i].str, diag->rsp_size[i]);
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
|
||||
free(diag);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int qtinfo_opt_p(int fd, char *support)
|
||||
{
|
||||
int ret;
|
||||
int sup = atoi(support);
|
||||
|
||||
ret = ioctl(fd, QTFS_IOCTL_EPOLL_SUPPORT, sup);
|
||||
if (ret != QTOK) {
|
||||
qtinfo_out("Set qtfs epoll support to:%s failed.", (sup == 1) ? "any file" : "fifo file");
|
||||
return ret;
|
||||
}
|
||||
qtinfo_out("Set qtfs epoll support to %s success.", (sup == 1) ? "any file" : "fifo file");
|
||||
return ret;
|
||||
}
|
||||
|
||||
#define PATH_MAX 4096
|
||||
static int qtinfo_opt_u()
|
||||
{
|
||||
int ret = -1;
|
||||
int len;
|
||||
struct sockaddr_un svr;
|
||||
char buf[RECV_BUFF_LEN];
|
||||
int sockfd = socket(AF_UNIX, SOCK_STREAM, 0);
|
||||
if (sockfd < 0) {
|
||||
qtinfo_err("Create socket fd failed.");
|
||||
return -1;
|
||||
}
|
||||
|
||||
memset(&svr, 0, sizeof(svr));
|
||||
svr.sun_family = AF_UNIX;
|
||||
strcpy(svr.sun_path, UDS_DIAG_ADDR);
|
||||
len = offsetof(struct sockaddr_un, sun_path) + strlen(svr.sun_path);
|
||||
if (connect(sockfd, (struct sockaddr *)&svr, len) < 0) {
|
||||
qtinfo_err("connect to %s failed.", UDS_DIAG_ADDR);
|
||||
close(sockfd);
|
||||
return -1;
|
||||
}
|
||||
while (1) {
|
||||
memset(buf, 0, RECV_BUFF_LEN);
|
||||
ret = recv(sockfd, buf, RECV_BUFF_LEN, 0);
|
||||
if (ret <= 0)
|
||||
break;
|
||||
qtinfo_out2("%s", buf);
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
close(sockfd);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int qtinfo_opt_s()
|
||||
{
|
||||
int ret = -1;
|
||||
int len;
|
||||
struct sockaddr_un svr;
|
||||
char buf[RECV_BUFF_LEN];
|
||||
int sockfd = socket(AF_UNIX, SOCK_STREAM, 0);
|
||||
if (sockfd < 0) {
|
||||
qtinfo_err("Create socket fd failed.");
|
||||
return -1;
|
||||
}
|
||||
|
||||
memset(&svr, 0, sizeof(svr));
|
||||
svr.sun_family = AF_UNIX;
|
||||
strcpy(svr.sun_path, UDS_LOGLEVEL_UPD);
|
||||
len = offsetof(struct sockaddr_un, sun_path) + strlen(svr.sun_path);
|
||||
if (connect(sockfd, (struct sockaddr *)&svr, len) < 0) {
|
||||
qtinfo_err("connect to %s failed.", UDS_LOGLEVEL_UPD);
|
||||
close(sockfd);
|
||||
return -1;
|
||||
}
|
||||
while (1) {
|
||||
memset(buf, 0, RECV_BUFF_LEN);
|
||||
ret = recv(sockfd, buf, RECV_BUFF_LEN, 0);
|
||||
if (ret <= 0)
|
||||
break;
|
||||
qtinfo_out2("%s", buf);
|
||||
}
|
||||
qtinfo_out2("\n");
|
||||
close(sockfd);
|
||||
return ret;
|
||||
|
||||
}
|
||||
|
||||
char wl_type_str[QTFS_WHITELIST_MAX][16] = {
|
||||
#ifdef server
|
||||
"Open",
|
||||
"Write",
|
||||
"Read",
|
||||
"Readdir",
|
||||
"Mkdir",
|
||||
"Rmdir",
|
||||
"Create",
|
||||
"Unlink",
|
||||
"Rename",
|
||||
"Setattr",
|
||||
"Setxattr",
|
||||
"Mount",
|
||||
"Kill",
|
||||
#endif
|
||||
"Udsconnect"
|
||||
};
|
||||
|
||||
int qtinfo_match_cap(char *cap)
|
||||
{
|
||||
if (cap == NULL)
|
||||
return -1;
|
||||
for (int type = 0; type < QTFS_WHITELIST_MAX; type++) {
|
||||
if (strcasecmp(wl_type_str[type], cap) == 0)
|
||||
return type;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
#define PATH_MAX 4096
|
||||
static int qtinfo_opt_x(int fd, char *path, char *cap)
|
||||
{
|
||||
int ret = -1;
|
||||
int index = 0;
|
||||
struct qtfs_wl_item head;
|
||||
ret = qtinfo_match_cap(cap);
|
||||
if (ret < 0) {
|
||||
qtinfo_err("White list add type:%s unknown.", cap);
|
||||
return -1;
|
||||
}
|
||||
head.type = ret;
|
||||
head.len = strlen(path);
|
||||
if (head.len >= PATH_MAX || head.len == 0) {
|
||||
qtinfo_err("White list add len:%u invalid", head.len);
|
||||
return -1;
|
||||
}
|
||||
head.path = path;
|
||||
ret = ioctl(fd, QTFS_IOCTL_WL_ADD, &head);
|
||||
if (ret != QTOK) {
|
||||
qtinfo_err("ioctl add white list failed");
|
||||
ret = -1;
|
||||
} else {
|
||||
qtinfo_out("successed to add white list item:%s successed", path);
|
||||
ret = 0;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int qtinfo_opt_y(int fd, char *index, char *cap)
|
||||
{
|
||||
struct qtfs_wl_item head;
|
||||
int ret = qtinfo_match_cap(cap);
|
||||
if (ret < 0) {
|
||||
qtinfo_err("White list delete type:%s unknown.", cap);
|
||||
return -1;
|
||||
}
|
||||
head.index = atoi(index);
|
||||
head.type = ret;
|
||||
head.path = NULL;
|
||||
head.len = 0;
|
||||
ret = ioctl(fd, QTFS_IOCTL_WL_DEL, &head);
|
||||
if (ret != QTOK) {
|
||||
qtinfo_err("failed to delete white list index:%d", head.index);
|
||||
return -1;
|
||||
} else {
|
||||
qtinfo_out("successed to delete white list index:%d", head.index);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void qtinfo_opt_z_bytype(int fd, unsigned int type)
|
||||
{
|
||||
int ret;
|
||||
char query[PATH_MAX];
|
||||
struct qtfs_wl_item head;
|
||||
head.path = query;
|
||||
head.type = type;
|
||||
qtinfo_out("Get qtfs <%s> white list:", wl_type_str[type]);
|
||||
for (unsigned int index = 0; index < QTFS_WL_MAX_NUM; index++) {
|
||||
memset(head.path, 0, PATH_MAX);
|
||||
head.index = index;
|
||||
ret = ioctl(fd, QTFS_IOCTL_WL_GET, &head);
|
||||
if (ret != QTOK)
|
||||
break;
|
||||
qtinfo_out(" [index:%u][path:%s]", index, head.path);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
static int qtinfo_opt_z(int fd, char *cap)
|
||||
{
|
||||
int ret = -1;
|
||||
int index = 0;
|
||||
ret = qtinfo_match_cap(cap);
|
||||
|
||||
if (ret >= 0 && ret < QTFS_WHITELIST_MAX) {
|
||||
qtinfo_opt_z_bytype(fd, ret);
|
||||
return 0;
|
||||
}
|
||||
for (int i = 0; i < QTFS_WHITELIST_MAX; i++) {
|
||||
qtinfo_opt_z_bytype(fd, i);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void qtinfo_help(char *exec)
|
||||
{
|
||||
qtinfo_out("Usage: %s [OPTION].", exec);
|
||||
qtinfo_out("Display qtfs client/server diagnostic information.");
|
||||
#ifndef QTINFO_RELEASE
|
||||
qtinfo_out(" -a, All diag info.");
|
||||
qtinfo_out(" -c, Clear all diag info.");
|
||||
qtinfo_out(" -l, Set log level(valid param: \"NONE\", \"ERROR\", \"WARN\", \"INFO\", \"DEBUG\").");
|
||||
qtinfo_out(" -t, For test informations.");
|
||||
qtinfo_out(" -p, Epoll support file mode(1: any files; 0: only fifo).");
|
||||
qtinfo_out(" -u, Display unix socket proxy diagnostic info");
|
||||
qtinfo_out(" -s, Set unix socket proxy log level(Increase by 1 each time)");
|
||||
#endif
|
||||
#ifdef server
|
||||
qtinfo_out(" -w, White list type(open/write/read/readdir/mkdir/rmdir/create/\n"
|
||||
" unlink/rename/setattr/setxattr/mount/kill/udsconnect)");
|
||||
#endif
|
||||
#ifdef client
|
||||
qtinfo_out(" -w, White list type(udsconnect)");
|
||||
#endif
|
||||
qtinfo_out(" -x, Add a qtfs white list path(example: -x /home/, must use with -w)");
|
||||
qtinfo_out(" -y, Delete a qtfs white list with index(example: -y 1, must use with -w)");
|
||||
qtinfo_out(" -z, Get all qtfs white list(just use -z, or with white list type)");
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
#define MAX_CAP_LEN 16
|
||||
int ret = -1;
|
||||
int ch;
|
||||
char wl_cap[MAX_CAP_LEN] = {0};
|
||||
if ((argc == 1) || (argc == 2 && strcmp(argv[1], "--help") == 0)) {
|
||||
qtinfo_help(argv[0]);
|
||||
return -1;
|
||||
}
|
||||
int fd = open(QTFS_DEV_NAME, O_RDONLY|O_NONBLOCK);
|
||||
if (fd < 0) {
|
||||
qtinfo_err("open file %s failed.", QTFS_DEV_NAME);
|
||||
#ifdef QTINFO_RELEASE
|
||||
return -1;
|
||||
#endif
|
||||
}
|
||||
#ifndef QTINFO_RELEASE
|
||||
while ((ch = getopt(argc, argv, "acl:tp:usw:x:y:z::")) != -1) {
|
||||
#else
|
||||
while ((ch = getopt(argc, argv, "w:x:y:z::")) != -1) {
|
||||
#endif
|
||||
switch (ch) {
|
||||
#ifndef QTINFO_RELEASE
|
||||
case 'a':
|
||||
ret = qtinfo_opt_a(fd);
|
||||
break;
|
||||
case 'c':
|
||||
ret = qtinfo_opt_c(fd);
|
||||
break;
|
||||
case 'l':
|
||||
ret = qtinfo_opt_l(fd, optarg);
|
||||
break;
|
||||
case 't':
|
||||
ret = qtinfo_opt_t(fd);
|
||||
break;
|
||||
case 'p':
|
||||
ret = qtinfo_opt_p(fd, optarg);
|
||||
break;
|
||||
case 'u':
|
||||
ret = qtinfo_opt_u();
|
||||
break;
|
||||
case 's':
|
||||
ret = qtinfo_opt_s();
|
||||
break;
|
||||
#endif
|
||||
case 'w':
|
||||
strncpy(wl_cap, optarg, MAX_CAP_LEN - 1);
|
||||
break;
|
||||
case 'x':
|
||||
ret = qtinfo_opt_x(fd, optarg, wl_cap);
|
||||
break;
|
||||
case 'y':
|
||||
ret = qtinfo_opt_y(fd, optarg, wl_cap);
|
||||
break;
|
||||
case 'z':
|
||||
ret = qtinfo_opt_z(fd, optarg);
|
||||
break;
|
||||
default:
|
||||
ret = 0;
|
||||
qtinfo_help(argv[0]);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
close(fd);
|
||||
return ret;
|
||||
}
|
78
qtfs/qtinfo/qtinfo.h
Normal file
78
qtfs/qtinfo/qtinfo.h
Normal file
@ -0,0 +1,78 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#ifndef __QTINFO_H__
|
||||
#define __QTINFO_H__
|
||||
|
||||
|
||||
enum qtfs_req_type
|
||||
{
|
||||
QTFS_REQ_NULL,
|
||||
QTFS_REQ_MOUNT,
|
||||
QTFS_REQ_OPEN,
|
||||
QTFS_REQ_CLOSE,
|
||||
QTFS_REQ_READ,
|
||||
QTFS_REQ_READITER, // 5
|
||||
QTFS_REQ_WRITE,
|
||||
QTFS_REQ_LOOKUP,
|
||||
QTFS_REQ_READDIR,
|
||||
QTFS_REQ_MKDIR,
|
||||
QTFS_REQ_RMDIR, // 10
|
||||
QTFS_REQ_GETATTR,
|
||||
QTFS_REQ_SETATTR,
|
||||
QTFS_REQ_ICREATE,
|
||||
QTFS_REQ_MKNOD,
|
||||
QTFS_REQ_UNLINK, // 15
|
||||
QTFS_REQ_SYMLINK,
|
||||
QTFS_REQ_LINK,
|
||||
QTFS_REQ_GETLINK,
|
||||
QTFS_REQ_READLINK,
|
||||
QTFS_REQ_RENAME, // 20
|
||||
|
||||
QTFS_REQ_XATTRLIST,
|
||||
QTFS_REQ_XATTRGET,
|
||||
QTFS_REQ_XATTRSET,
|
||||
|
||||
QTFS_REQ_SYSMOUNT,
|
||||
QTFS_REQ_SYSUMOUNT, // 25
|
||||
QTFS_REQ_FIFOPOLL,
|
||||
|
||||
QTFS_REQ_STATFS,
|
||||
QTFS_REQ_IOCTL,
|
||||
|
||||
QTFS_REQ_EPOLL_CTL,
|
||||
|
||||
QTFS_REQ_EPOLL_EVENT,
|
||||
|
||||
QTFS_REQ_LLSEEK,
|
||||
|
||||
// REMOTE SYSCALL
|
||||
QTFS_SC_KILL,
|
||||
QTFS_SC_SCHED_GETAFFINITY,
|
||||
QTFS_SC_SCHED_SETAFFINITY,
|
||||
|
||||
QTFS_REQ_EXIT,
|
||||
QTFS_REQ_INV,
|
||||
};
|
||||
|
||||
#define MAX_QTINFO_TYPE_STR_LEN 20
|
||||
|
||||
struct qtinfo_type_str {
|
||||
enum qtfs_req_type type;
|
||||
char str[MAX_QTINFO_TYPE_STR_LEN];
|
||||
};
|
||||
|
||||
#endif
|
||||
|
26
qtfs/rexec/Makefile
Normal file
26
qtfs/rexec/Makefile
Normal file
@ -0,0 +1,26 @@
|
||||
DEPGLIB=-lglib-2.0 -I/usr/include/glib-2.0 -I/usr/lib64/glib-2.0/include
|
||||
|
||||
CFLAGS += -g -O2
|
||||
CFLAGS += -fstack-protector-strong
|
||||
CFLAGS += -fPIE -pie -fPIC
|
||||
CFLAGS += -D_FORTIFY_SOURCE=2
|
||||
LDFLAGS += -Wl,-z,now
|
||||
LDFLAGS += -Wl,-z,noexecstack
|
||||
LDFLAGS += -fPIE -pie
|
||||
|
||||
all: rexec rexec_server
|
||||
|
||||
rexec :
|
||||
gcc $(CFLAGS) $(LDFLAGS) -o rexec rexec.c rexec_sock.c -ljson-c
|
||||
|
||||
rexec_server :
|
||||
gcc $(CFLAGS) $(LDFLAGS) -o rexec_server rexec_server.c rexec_sock.c rexec_shim.c -ljson-c $(DEPGLIB)
|
||||
test:
|
||||
go test -v ./common_test.go ./common.go
|
||||
|
||||
install:
|
||||
yes | cp -f rexec /usr/bin/
|
||||
yes | cp -f rexec_server /usr/bin/
|
||||
|
||||
clean:
|
||||
rm -rf rexec rexec_server
|
36
qtfs/rexec/README.md
Normal file
36
qtfs/rexec/README.md
Normal file
@ -0,0 +1,36 @@
|
||||
# rexec
|
||||
|
||||
rexec is a sample application demonstrating libchan nested channels and
|
||||
bytestreams through remote executation calls. This is a minimal
|
||||
implementation to demonstrate the usage of libchan.
|
||||
|
||||
### Usage
|
||||
|
||||
Server
|
||||
~~~~
|
||||
$ cd rexec_server
|
||||
$ go build .
|
||||
$ ./rexec_server
|
||||
~~~~
|
||||
|
||||
Client
|
||||
~~~~
|
||||
$ go build .
|
||||
$ ./rexec /bin/echo "hello"
|
||||
hello
|
||||
$ ./rexec /bin/sh -c "exit 4"
|
||||
$ echo $?
|
||||
4
|
||||
~~~~
|
||||
|
||||
### Usage with TLS
|
||||
Server
|
||||
~~~~
|
||||
$ TLS_CERT=./cert.pem TLS_KEY=./key.pem ./rexec_server
|
||||
~~~~
|
||||
|
||||
Client
|
||||
~~~~
|
||||
$ USE_TLS=true ./rexec /bin/echo "hello"
|
||||
hello
|
||||
~~~~
|
773
qtfs/rexec/rexec.c
Normal file
773
qtfs/rexec/rexec.c
Normal file
@ -0,0 +1,773 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
#include <unistd.h>
|
||||
#include <signal.h>
|
||||
#include <pthread.h>
|
||||
#include <sys/epoll.h>
|
||||
#include <netinet/ip.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/un.h>
|
||||
#include <netinet/udp.h>
|
||||
#include <netinet/tcp.h>
|
||||
#include <sys/socket.h>
|
||||
#include <arpa/inet.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/file.h>
|
||||
#include <json-c/json_object.h>
|
||||
|
||||
#include "dirent.h"
|
||||
|
||||
#include "rexec_sock.h"
|
||||
#include "rexec.h"
|
||||
|
||||
#define REXEC_MSG_LEN 1024
|
||||
FILE *rexec_logfile = NULL;
|
||||
|
||||
struct rexec_global_var {
|
||||
int rexec_hs_fd[2];
|
||||
};
|
||||
|
||||
struct rexec_thread_arg {
|
||||
int efd;
|
||||
int connfd;
|
||||
char **argv;
|
||||
};
|
||||
|
||||
struct rexec_global_var g_rexec;
|
||||
|
||||
|
||||
struct rexec_client_event {
|
||||
int fd;
|
||||
int outfd; // for stdin out err and other pipe
|
||||
int (*handler)(struct rexec_client_event *);
|
||||
int *exit_status;
|
||||
int *pidfd;
|
||||
};
|
||||
|
||||
#define REXEC_PIDMAP_PATH "/var/run/rexec/pids"
|
||||
#define REXEC_PIDMAP_PATH_LEN 64
|
||||
#define REXEC_PID_LEN 16
|
||||
static int rexec_conn_to_server()
|
||||
{
|
||||
struct rexec_conn_arg arg;
|
||||
char *ret = strncpy(arg.sun_path, REXEC_UDS_CONN, sizeof(arg.sun_path));
|
||||
if (ret == NULL) {
|
||||
rexec_err("strncpy sun path failed");
|
||||
return -1;
|
||||
}
|
||||
arg.cs = REXEC_SOCK_CLIENT;
|
||||
arg.udstype = SOCK_STREAM;
|
||||
if (rexec_build_unix_connection(&arg) != 0)
|
||||
return -1;
|
||||
return arg.connfd;
|
||||
}
|
||||
|
||||
static int rexec_calc_argv_len(int argc, char *argv[])
|
||||
{
|
||||
int len = 0;
|
||||
for (int i = 0; i < argc; i++) {
|
||||
if (argv[i] == NULL) {
|
||||
rexec_err("Invalid argv index:%d", i);
|
||||
return -1;
|
||||
}
|
||||
len += strlen(argv[i]);
|
||||
len++;
|
||||
}
|
||||
return len;
|
||||
}
|
||||
|
||||
static int rexec_msg_fill_argv(int argc, char *argv[], char *msg)
|
||||
{
|
||||
int offset = 0;
|
||||
for (int i = 0; i < argc; i++) {
|
||||
strcpy(&msg[offset], argv[i]); //此处msg已经在前面通过计算出的len预先分配内存,保证这里不会越界
|
||||
offset += (strlen(argv[i]) + 1);
|
||||
}
|
||||
return offset;
|
||||
}
|
||||
|
||||
static int rexec_io(struct rexec_client_event *evt)
|
||||
{
|
||||
#define MAX_MSG_LEN 256
|
||||
char buf[MAX_MSG_LEN];
|
||||
int len;
|
||||
int ret;
|
||||
while ((len = read(evt->fd, buf, MAX_MSG_LEN)) > 0) {
|
||||
ret = write(evt->outfd, buf, len);
|
||||
if (ret <= 0) {
|
||||
rexec_err("Read from fd:%d len:%d write to fd:%d failed ret:%d", evt->fd, len, evt->outfd, ret);
|
||||
return REXEC_EVENT_DEL;
|
||||
}
|
||||
if (ret != len) {
|
||||
rexec_err("Read from fd:%d len:%d but write to fd:%d ret:%d", evt->fd, len, evt->outfd, ret);
|
||||
}
|
||||
}
|
||||
return REXEC_EVENT_OK;
|
||||
}
|
||||
|
||||
// return -1 means process exit.
|
||||
static int rexec_conn_msg(struct rexec_client_event *evt)
|
||||
{
|
||||
struct rexec_msg head;
|
||||
int ret = recv(evt->fd, &head, sizeof(struct rexec_msg), MSG_WAITALL);
|
||||
if (ret <= 0) {
|
||||
rexec_err("Rexec conn recv err:%d errno:%d", ret, errno);
|
||||
return REXEC_EVENT_EXIT;
|
||||
}
|
||||
switch (head.msgtype) {
|
||||
case REXEC_KILL:
|
||||
*evt->exit_status = head.exit_status;
|
||||
rexec_err("Rexec conn recv kill msg, exit:%d now.", head.exit_status);
|
||||
return REXEC_EVENT_EXIT;
|
||||
case REXEC_PIDMAP: {
|
||||
int mypid = getpid();
|
||||
int peerpid = head.pid;
|
||||
char path[REXEC_PIDMAP_PATH_LEN] = {0};
|
||||
char buf[REXEC_PID_LEN] = {0};
|
||||
int fd;
|
||||
int err;
|
||||
if (*evt->pidfd > 0) {
|
||||
rexec_err("Rexec pidmap msg > 1 error.");
|
||||
return REXEC_EVENT_OK;
|
||||
}
|
||||
sprintf(path, "%s/%d", REXEC_PIDMAP_PATH, mypid);
|
||||
fd = open(path, O_CREAT|O_WRONLY, 0600);
|
||||
if (fd < 0) {
|
||||
rexec_err("Rexec create pidmap:%d-%d failed, path:%s open failed:%d",
|
||||
mypid, peerpid, path, fd);
|
||||
break;
|
||||
}
|
||||
*evt->pidfd = fd;
|
||||
if ((err = flock(fd, LOCK_EX)) != 0) {
|
||||
rexec_err("Rexec flock file:%s failed, errno:%d rexec exit.", path, err);
|
||||
return REXEC_EVENT_EXIT;
|
||||
}
|
||||
if ((err = ftruncate(fd, 0)) != 0) {
|
||||
rexec_err("Rexec pidmap file:%s clear failed errno:%d rexec exit.", path, err);
|
||||
return REXEC_EVENT_EXIT;
|
||||
}
|
||||
if ((err = lseek(fd, 0, SEEK_SET)) < 0) {
|
||||
rexec_err("Rexec pidmap file:%s lseek 0 failed errno:%d rexec exit", path, err);
|
||||
return REXEC_EVENT_EXIT;
|
||||
}
|
||||
sprintf(buf, "%d", peerpid);
|
||||
if ((err = write(fd, buf, strlen(buf))) <= 0) {
|
||||
rexec_err("Rexec pidmap file:%s write pid:%d failed errno:%d rexec exit.", path, peerpid, err);
|
||||
return REXEC_EVENT_EXIT;
|
||||
}
|
||||
if (g_rexec.rexec_hs_fd[PIPE_WRITE] != -1 && g_rexec.rexec_hs_fd[PIPE_READ] != -1) {
|
||||
err = write(g_rexec.rexec_hs_fd[PIPE_WRITE], "1", 1);
|
||||
if (err <= 0) {
|
||||
rexec_err("rexec handshake write 1 failed, hs write:%d.", g_rexec.rexec_hs_fd[PIPE_WRITE]);
|
||||
return REXEC_EVENT_ERR;
|
||||
}
|
||||
} else {
|
||||
char msg[sizeof(struct rexec_msg) + 1];
|
||||
struct rexec_msg *hs = msg;
|
||||
char *ok = hs->msg;
|
||||
hs->msgtype = REXEC_HANDSHAKE;
|
||||
hs->msglen = 1;
|
||||
*ok = '1';
|
||||
if (write(evt->fd, hs, sizeof(struct rexec_msg) + 1) <= 0) {
|
||||
rexec_err("send handshake failed, remote process will die");
|
||||
return REXEC_EVENT_EXIT;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
rexec_log("Rexec conn recv msgtype:%d argc:%d pipefd:%d msglen:%d",
|
||||
head.msgtype, head.argc, head.pipefd, head.msglen);
|
||||
return REXEC_EVENT_OK;
|
||||
}
|
||||
|
||||
static struct rexec_client_event *rexec_add_event(int efd, int fd, int outfd, int (*handler)(struct rexec_client_event *))
|
||||
{
|
||||
struct rexec_client_event *event = (struct rexec_client_event *)malloc(sizeof(struct rexec_client_event));
|
||||
if (event == NULL) {
|
||||
rexec_err("malloc failed.");
|
||||
return NULL;
|
||||
}
|
||||
event->fd = fd;
|
||||
event->outfd = outfd;
|
||||
event->handler = handler;
|
||||
struct epoll_event evt;
|
||||
evt.data.ptr = (void *)event;
|
||||
evt.events = EPOLLIN;
|
||||
if (-1 == epoll_ctl(efd, EPOLL_CTL_ADD, event->fd, &evt)) {
|
||||
rexec_err("epoll ctl add fd:%d event failed.", event->fd);
|
||||
free(event);
|
||||
return NULL;
|
||||
}
|
||||
return event;
|
||||
}
|
||||
|
||||
static int rexec_del_event(struct rexec_client_event *event)
|
||||
{
|
||||
// close will del fd in epoll list
|
||||
close(event->fd);
|
||||
free(event);
|
||||
return 0;
|
||||
}
|
||||
|
||||
enum {
|
||||
REPOL_IN_INDEX = 0,
|
||||
REPOL_OUT_INDEX,
|
||||
REPOL_ERR_INDEX,
|
||||
REPOL_INV_INDEX,
|
||||
};
|
||||
static int rexec_std_event(int efd, int rstdin, int rstdout, int rstderr)
|
||||
{
|
||||
#define REXEC_MAX_EVENTS 4
|
||||
int infds[REPOL_INV_INDEX] = {STDIN_FILENO, rstdout, rstderr};
|
||||
int outfds[REPOL_INV_INDEX] = {rstdin, STDOUT_FILENO, STDERR_FILENO};
|
||||
|
||||
for (int i = 0; i < REPOL_INV_INDEX; i++) {
|
||||
if (NULL == rexec_add_event(efd, infds[i], outfds[i], rexec_io)) {
|
||||
rexec_err("epoll ctl add fd:%d event failed and ignore this mistake.", infds[i]);
|
||||
continue;
|
||||
} else {
|
||||
if (rexec_set_nonblock(infds[i], 1) != 0) {
|
||||
rexec_err("rexec set fd:%d i:%d non block failed.", infds[i], i);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void rexec_event_run(int efd)
|
||||
{
|
||||
struct epoll_event *evts = calloc(REXEC_MAX_EVENTS, sizeof(struct epoll_event));
|
||||
if (evts == NULL) {
|
||||
rexec_err("init calloc evts failed.");
|
||||
return;
|
||||
}
|
||||
while (1) {
|
||||
int n = epoll_wait(efd, evts, REXEC_MAX_EVENTS, 1000);
|
||||
int process_exit = 0;
|
||||
if (n == 0)
|
||||
continue;
|
||||
if (n < 0) {
|
||||
rexec_err("epoll wait return errcode:%d", n);
|
||||
continue;
|
||||
}
|
||||
for (int i = 0; i < n; i++) {
|
||||
struct rexec_client_event *evt = (struct rexec_client_event *)evts[i].data.ptr;
|
||||
int ret = evt->handler(evt);
|
||||
if (ret == REXEC_EVENT_EXIT) {
|
||||
process_exit = 1;
|
||||
}
|
||||
if (ret == REXEC_EVENT_DEL) {
|
||||
rexec_del_event(evt);
|
||||
}
|
||||
}
|
||||
// process will exit, and free all resource and exit
|
||||
if (process_exit) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
free(evts);
|
||||
return;
|
||||
}
|
||||
|
||||
static int rexec_run(int efd, int connfd, char *argv[])
|
||||
{
|
||||
int pidfd = -1;
|
||||
int exit_status = EXIT_FAILURE;
|
||||
|
||||
struct rexec_client_event *connevt = rexec_add_event(efd, connfd, -1, rexec_conn_msg);
|
||||
if (NULL == connevt || rexec_set_nonblock(connfd, 1) != 0) {
|
||||
// process will exit, fd or mem resource will free by kernel soon
|
||||
rexec_err("rexec add connfd event failed");
|
||||
return exit_status;
|
||||
}
|
||||
// 这两个指针只能在当前函数上下文使用,是当前函数栈指针
|
||||
connevt->exit_status = &exit_status;
|
||||
connevt->pidfd = &pidfd;
|
||||
|
||||
rexec_log("Rexec process start run, as proxy of remote %s", argv[1]);
|
||||
rexec_event_run(efd);
|
||||
rexec_log("Rexec process %s exit.", argv[1]);
|
||||
|
||||
// clear pidmap file
|
||||
if (pidfd > 0) {
|
||||
char path[32] = {0};
|
||||
sprintf(path, "%s/%d", REXEC_PIDMAP_PATH, getpid());
|
||||
close(pidfd);
|
||||
remove(path);
|
||||
}
|
||||
|
||||
end:
|
||||
close(efd);
|
||||
return exit_status;
|
||||
}
|
||||
|
||||
void rexec_create_pidmap_path()
|
||||
{
|
||||
if (access(REXEC_RUN_PATH, F_OK) != 0) {
|
||||
mkdir(REXEC_RUN_PATH, 0700);
|
||||
}
|
||||
mkdir(REXEC_PIDMAP_PATH, 0700);
|
||||
return;
|
||||
}
|
||||
|
||||
void rexec_clear_pids()
|
||||
{
|
||||
char path[REXEC_PIDMAP_PATH_LEN] = {0};
|
||||
DIR *dir = NULL;
|
||||
struct dirent *entry;
|
||||
if (access(REXEC_PIDMAP_PATH, F_OK) != 0) {
|
||||
rexec_create_pidmap_path();
|
||||
return;
|
||||
}
|
||||
dir = opendir(REXEC_PIDMAP_PATH);
|
||||
if (dir == NULL) {
|
||||
rexec_err("open path:%s failed", REXEC_PIDMAP_PATH);
|
||||
return;
|
||||
}
|
||||
while (entry = readdir(dir)) {
|
||||
int fd;
|
||||
|
||||
if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0 ||
|
||||
strlen(entry->d_name) >= REXEC_PID_LEN)
|
||||
continue;
|
||||
|
||||
memset(path, 0, sizeof(path));
|
||||
sprintf(path, "%s/%s", REXEC_PIDMAP_PATH, entry->d_name);
|
||||
fd = open(path, O_RDONLY);
|
||||
if (fd <= 0) {
|
||||
rexec_err("open pid file:%s failed", path);
|
||||
continue;
|
||||
}
|
||||
if (flock(fd, LOCK_EX|LOCK_NB) != 0) {
|
||||
close(fd);
|
||||
continue;
|
||||
}
|
||||
close(fd);
|
||||
if (remove(path) != 0) {
|
||||
rexec_err("remove unuse pidmap file:%s failed", path);
|
||||
}
|
||||
}
|
||||
|
||||
closedir(dir);
|
||||
return;
|
||||
}
|
||||
|
||||
#define REXEC_PATH_MAX 4096
|
||||
struct rexec_fdinfo {
|
||||
int fd;
|
||||
char path[REXEC_PATH_MAX];
|
||||
unsigned int perm;
|
||||
int offset;
|
||||
};
|
||||
|
||||
static inline int rexec_is_reg_file(int fd)
|
||||
{
|
||||
if (S_ISREG(rexec_fd_mode(fd)))
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rexec_get_fdinfo(struct dirent *fdentry, struct rexec_fdinfo *fdinfo)
|
||||
{
|
||||
char path[32] = {0};
|
||||
int ret;
|
||||
int fd = atoi(fdentry->d_name);
|
||||
if (fd <= STDERR_FILENO || fd == fileno(rexec_logfile))
|
||||
return -1;
|
||||
if (!rexec_is_reg_file(fd))
|
||||
return -1;
|
||||
sprintf(path, "/proc/self/fd/%s", fdentry->d_name);
|
||||
ret = readlink(path, fdinfo->path, REXEC_PATH_MAX);
|
||||
if (ret < 0) {
|
||||
rexec_err("Get fd:%d link failed.", fd);
|
||||
return -1;
|
||||
}
|
||||
fdinfo->fd = fd;
|
||||
fdinfo->offset = lseek(fd, 0, SEEK_CUR);
|
||||
|
||||
fdinfo->perm = fcntl(fd, F_GETFL, NULL);
|
||||
if (fdinfo->perm == -1) {
|
||||
rexec_err("Get fd:%d flags failed", fd);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
// 返回一个拼装好的json格式字符串,内存在内部申请好
|
||||
// 由调用者释放
|
||||
// 内容是本进程所有REG类型文件的信息
|
||||
static char *rexec_get_fds_jsonstr()
|
||||
{
|
||||
struct json_object *root = json_object_new_object();
|
||||
char *json_str;
|
||||
int len;
|
||||
DIR *fddir = NULL;
|
||||
struct dirent *fdentry;
|
||||
struct rexec_fdinfo *fdinfo;
|
||||
if (root == NULL) {
|
||||
rexec_err("create json-c root failed.");
|
||||
return NULL;
|
||||
}
|
||||
fdinfo = (struct rexec_fdinfo *)malloc(sizeof(struct rexec_fdinfo));
|
||||
if (fdinfo == NULL) {
|
||||
rexec_err("malloc failed.");
|
||||
goto err_end;
|
||||
}
|
||||
|
||||
fddir = opendir("/proc/self/fd");
|
||||
if (fddir == NULL) {
|
||||
free(fdinfo);
|
||||
rexec_err("open path:/proc/self/fd failed");
|
||||
goto err_end;
|
||||
}
|
||||
|
||||
struct json_object *files_arr = json_object_new_array();
|
||||
|
||||
while (fdentry = readdir(fddir)) {
|
||||
struct json_object *fd_obj = json_object_new_object();
|
||||
struct json_object *item = NULL;
|
||||
|
||||
if (fd_obj == NULL) {
|
||||
rexec_err("json c new object failed.");
|
||||
goto json_err;
|
||||
}
|
||||
memset(fdinfo, 0, sizeof(struct rexec_fdinfo));
|
||||
if (rexec_get_fdinfo(fdentry, fdinfo) != 0) {
|
||||
json_object_put(fd_obj);
|
||||
continue;
|
||||
}
|
||||
item = json_object_new_int(fdinfo->fd);
|
||||
json_object_object_add(fd_obj, "Fd", item);
|
||||
item = json_object_new_string(fdinfo->path);
|
||||
json_object_object_add(fd_obj, "Path", item);
|
||||
item = json_object_new_int(fdinfo->perm);
|
||||
json_object_object_add(fd_obj, "Perm", item);
|
||||
item = json_object_new_int(fdinfo->offset);
|
||||
json_object_object_add(fd_obj, "Offset", item);
|
||||
|
||||
json_object_array_add(files_arr, fd_obj);
|
||||
}
|
||||
closedir(fddir);
|
||||
free(fdinfo);
|
||||
|
||||
json_object_object_add(root, "Files", files_arr);
|
||||
json_str = strdup(json_object_get_string(root));
|
||||
json_object_put(root);
|
||||
|
||||
return json_str;
|
||||
|
||||
json_err:
|
||||
closedir(fddir);
|
||||
free(fdinfo);
|
||||
err_end:
|
||||
json_object_put(root);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
// 将rexec进程从parent继承到的匿名pipe继承给远端进程
|
||||
static int rexec_pipe_remote_inherit(int efd, int connfd)
|
||||
{
|
||||
#define SELF_FD_PATH "/proc/self/fd"
|
||||
DIR *fddir = NULL;
|
||||
struct dirent *fdentry;
|
||||
struct rexec_msg msg;
|
||||
mode_t mode;
|
||||
int pfd[2];
|
||||
|
||||
fddir = opendir(SELF_FD_PATH);
|
||||
if (fddir == NULL) {
|
||||
rexec_err("open path:%s failed", SELF_FD_PATH);
|
||||
return -1;
|
||||
}
|
||||
memset(&msg, 0, sizeof(struct rexec_msg));
|
||||
msg.msglen = 0;
|
||||
msg.pipefd = -1;
|
||||
msg.msgtype = REXEC_PIPE;
|
||||
while (fdentry = readdir(fddir)) {
|
||||
int fd = atoi(fdentry->d_name);
|
||||
if (fd <= STDERR_FILENO)
|
||||
continue;
|
||||
mode = rexec_fd_mode(fd);
|
||||
if (!S_ISFIFO(mode))
|
||||
continue;
|
||||
rexec_log("inherit pipe fd:%d mode:%o is %s pipe", fd, mode, (!!(mode & S_IRUSR)) ? "read" : "write");
|
||||
if (pipe(pfd) == -1) {
|
||||
rexec_err("failed to create pipe for:%d", fd);
|
||||
goto err_end;
|
||||
}
|
||||
msg.pipefd = fd;
|
||||
if (!!(mode & S_IRUSR)) {
|
||||
// inherit read pipe
|
||||
if (rexec_sendmsg(connfd, (char *)&msg, sizeof(struct rexec_msg), pfd[PIPE_READ]) < 0) {
|
||||
rexec_err("send read pipe failed, inherit fd:%d", fd);
|
||||
goto pipe_end;
|
||||
}
|
||||
if (rexec_add_event(efd, fd, pfd[PIPE_WRITE], rexec_io) == NULL) {
|
||||
rexec_err("add read pipe event failed:%d", fd);
|
||||
goto pipe_end;
|
||||
}
|
||||
close(pfd[PIPE_READ]);
|
||||
} else if (!!(mode & S_IWUSR)) {
|
||||
if (rexec_sendmsg(connfd, (char *)&msg, sizeof(struct rexec_msg), pfd[PIPE_WRITE]) < 0) {
|
||||
rexec_err("send write pipe failed, inherit fd:%d", fd);
|
||||
goto pipe_end;
|
||||
}
|
||||
if (rexec_add_event(efd, pfd[PIPE_READ], fd, rexec_io) == NULL) {
|
||||
rexec_err("add write pipe event failed:%d", fd);
|
||||
goto pipe_end;
|
||||
}
|
||||
close(pfd[PIPE_WRITE]);
|
||||
}
|
||||
rexec_log("successed to add pipe fd:%d to remote inherit", fd);
|
||||
}
|
||||
closedir(fddir);
|
||||
return 0;
|
||||
|
||||
pipe_end:
|
||||
close(pfd[0]);
|
||||
close(pfd[1]);
|
||||
err_end:
|
||||
closedir(fddir);
|
||||
return -1;
|
||||
}
|
||||
|
||||
static int rexec_handshake_proc(struct rexec_client_event *evt)
|
||||
{
|
||||
char msg[sizeof(struct rexec_msg) + 1];
|
||||
struct rexec_msg *hs = msg;
|
||||
int ret = read(evt->fd, hs->msg, 1);
|
||||
if (ret <= 0) {
|
||||
rexec_err("read from handshake pipe failed, ret:%d err:%d", ret, errno);
|
||||
return REXEC_EVENT_DEL;
|
||||
}
|
||||
hs->msgtype = REXEC_HANDSHAKE;
|
||||
hs->msglen = 1;
|
||||
ret = write(evt->outfd, hs, sizeof(struct rexec_msg) + 1);
|
||||
if (ret < 0) {
|
||||
rexec_err("send handshake failed, connfd:%d.", evt->outfd);
|
||||
}
|
||||
return REXEC_EVENT_OK;
|
||||
}
|
||||
|
||||
static int rexec_handshake_init(int efd, int connfd)
|
||||
{
|
||||
char *hs_read = getenv("REXEC_HANDSHAKE_RD");
|
||||
if (hs_read == NULL) {
|
||||
rexec_log("handshake not in effect.");
|
||||
return 0;
|
||||
}
|
||||
g_rexec.rexec_hs_fd[PIPE_READ] = atoi(hs_read);
|
||||
|
||||
char *hs_write = getenv("REXEC_HANDSHAKE_WR");
|
||||
if (hs_write == NULL) {
|
||||
rexec_log("handshake not in effect, read:%s", hs_read);
|
||||
g_rexec.rexec_hs_fd[PIPE_READ] = -1;
|
||||
return 0;
|
||||
}
|
||||
g_rexec.rexec_hs_fd[PIPE_WRITE] = atoi(hs_write);
|
||||
|
||||
if (g_rexec.rexec_hs_fd[PIPE_READ] <= STDERR_FILENO || g_rexec.rexec_hs_fd[PIPE_WRITE] <= STDERR_FILENO) {
|
||||
rexec_log("handshake invalid fd read:%d write:%d", g_rexec.rexec_hs_fd[PIPE_READ], g_rexec.rexec_hs_fd[PIPE_WRITE]);
|
||||
goto err_end;
|
||||
}
|
||||
if (!S_ISFIFO(rexec_fd_mode(g_rexec.rexec_hs_fd[PIPE_READ])) || !S_ISFIFO(rexec_fd_mode(g_rexec.rexec_hs_fd[PIPE_WRITE]))) {
|
||||
rexec_err("handshake fd mode not fifo:%d %d", g_rexec.rexec_hs_fd[PIPE_READ], g_rexec.rexec_hs_fd[PIPE_WRITE]);
|
||||
goto err_end;
|
||||
}
|
||||
if (rexec_add_event(efd, g_rexec.rexec_hs_fd[PIPE_READ], connfd, rexec_handshake_proc) == NULL) {
|
||||
rexec_err("add handshake pipe read fd:%d to epoll failed", g_rexec.rexec_hs_fd[PIPE_READ]);
|
||||
goto err_end;
|
||||
}
|
||||
rexec_log("handshake effect read:%d write:%d", g_rexec.rexec_hs_fd[PIPE_READ], g_rexec.rexec_hs_fd[PIPE_WRITE]);
|
||||
return 0;
|
||||
err_end:
|
||||
g_rexec.rexec_hs_fd[PIPE_READ] = -1;
|
||||
g_rexec.rexec_hs_fd[PIPE_WRITE] = -1;
|
||||
return -1;
|
||||
}
|
||||
|
||||
static int rexec_send_binary_msg(int efd, int argc, char *argv[], int arglen, char *fds_json, int connfd)
|
||||
{
|
||||
struct rexec_msg *pmsg = (struct rexec_msg *)malloc(arglen);
|
||||
if (pmsg == NULL) {
|
||||
rexec_err("malloc failed");
|
||||
free(fds_json);
|
||||
return -1;
|
||||
}
|
||||
char *bufmsg = pmsg->msg;
|
||||
memset(pmsg, 0, arglen);
|
||||
pmsg->msgtype = REXEC_EXEC;
|
||||
pmsg->argc = argc - 1; // for remote binary's argc is argc-1
|
||||
// pmsg->msg is like: "binary"\0"argv[1]"\0"argv[2]"\0"..."
|
||||
pmsg->msglen = rexec_msg_fill_argv(pmsg->argc, &argv[1], bufmsg);
|
||||
strcpy(&bufmsg[pmsg->msglen], fds_json);
|
||||
pmsg->msglen += strlen(fds_json);
|
||||
free(fds_json);
|
||||
|
||||
// pipefd[0] -- for read; pipefd[1] -- for write.
|
||||
// rexec stdin --> rstdin[1] ------> rstdin[0] as stdin
|
||||
// rexec stdout <-- rstdout[0] <------ rstdout[1] as stdout
|
||||
// rexec stderr <-- rstderr[0] <------ rstderr[1] as stderr
|
||||
int rstdin[2];
|
||||
int rstdout[2];
|
||||
int rstderr[2];
|
||||
|
||||
if (pipe(rstdin) == -1 || pipe(rstdout) == -1 || pipe(rstderr) == -1) {
|
||||
rexec_err("Rexec create pipe failed.");
|
||||
goto err_end;
|
||||
}
|
||||
pmsg->pipefd = REXEC_STDIN;
|
||||
if (rexec_sendmsg(connfd, (char *)pmsg, sizeof(struct rexec_msg) + pmsg->msglen, rstdin[0]) < 0) {
|
||||
rexec_err("Rexec send exec msg failed, errno:%d", errno);
|
||||
goto err_end;
|
||||
}
|
||||
rexec_log("Normal msg send len:%d head:%d", sizeof(struct rexec_msg) + pmsg->msglen, sizeof(struct rexec_msg));
|
||||
pmsg->msgtype = REXEC_PIPE;
|
||||
pmsg->argc = 0;
|
||||
pmsg->msglen = 0;
|
||||
pmsg->pipefd = REXEC_STDOUT;
|
||||
if (rexec_sendmsg(connfd, (char *)pmsg, sizeof(struct rexec_msg), rstdout[1]) < 0) {
|
||||
rexec_err("Rexec send exec msg failed, errno:%d", errno);
|
||||
goto err_end;
|
||||
}
|
||||
pmsg->pipefd = REXEC_STDERR;
|
||||
if (rexec_sendmsg(connfd, (char *)pmsg, sizeof(struct rexec_msg), rstderr[1]) < 0) {
|
||||
rexec_err("Rexec send exec msg failed, errno:%d", errno);
|
||||
goto err_end;
|
||||
}
|
||||
|
||||
if (rexec_std_event(efd, rstdin[1], rstdout[0], rstderr[0]) != 0) {
|
||||
rexec_err("add std event failed");
|
||||
goto err_end;
|
||||
}
|
||||
free(pmsg);
|
||||
close(rstdin[0]);
|
||||
close(rstdout[1]);
|
||||
close(rstderr[1]);
|
||||
return 0;
|
||||
err_end:
|
||||
free(pmsg);
|
||||
return -1;
|
||||
}
|
||||
|
||||
static void *rexec_pipe_proxy_thread(void *arg)
|
||||
{
|
||||
struct rexec_thread_arg *parg = (struct rexec_thread_arg *)arg;
|
||||
rexec_log("pipe proxy thread run.");
|
||||
rexec_event_run(parg->efd);
|
||||
rexec_log("pipe proxy thread run over");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void *rexec_conn_thread(void *arg)
|
||||
{
|
||||
struct rexec_thread_arg *parg = (struct rexec_thread_arg *)arg;
|
||||
|
||||
return (void *)rexec_run(parg->efd, parg->connfd, parg->argv);
|
||||
}
|
||||
|
||||
static void rexec_global_var_init()
|
||||
{
|
||||
memset(&g_rexec, 0, sizeof(g_rexec));
|
||||
g_rexec.rexec_hs_fd[PIPE_READ] = -1;
|
||||
g_rexec.rexec_hs_fd[PIPE_WRITE] = -1;
|
||||
return;
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
rexec_log_init();
|
||||
rexec_clear_pids();
|
||||
|
||||
int pipeefd = epoll_create1(0);
|
||||
int efd = epoll_create1(0);
|
||||
if (efd == -1 || pipeefd == -1) {
|
||||
rexec_err("epoll create1 failed, errno:%d.", errno);
|
||||
return -1;
|
||||
}
|
||||
rexec_global_var_init();
|
||||
|
||||
int connfd = rexec_conn_to_server();
|
||||
if (connfd < 0) {
|
||||
rexec_err("Rexec connect to server failed, errno:%d", errno);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (rexec_handshake_init(efd, connfd) != 0) {
|
||||
rexec_err("Rexec handshake environment set but get error.");
|
||||
return -1;
|
||||
}
|
||||
rexec_log("Remote exec binary:%s", argv[1]);
|
||||
/*if (rexec_pipe_remote_inherit(pipeefd, connfd) != 0) {
|
||||
rexec_err("Rexec pipe remote inherit failed.");
|
||||
goto err_end;
|
||||
}*/
|
||||
|
||||
int arglen = rexec_calc_argv_len(argc - 1, &argv[1]);
|
||||
if (arglen <= 0) {
|
||||
rexec_err("argv is invalid.");
|
||||
return -1;
|
||||
}
|
||||
char *fds_json = rexec_get_fds_jsonstr();
|
||||
if (fds_json == NULL) {
|
||||
rexec_err("Get fds info json string failed.");
|
||||
return -1;
|
||||
}
|
||||
arglen += sizeof(struct rexec_msg);
|
||||
arglen += strlen(fds_json);
|
||||
arglen = ((arglen / REXEC_MSG_LEN) + 1) * REXEC_MSG_LEN;
|
||||
if (arglen <= 0) {
|
||||
rexec_err("invalid arguments length:%d.", arglen);
|
||||
free(fds_json);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (rexec_send_binary_msg(efd, argc, argv, arglen, fds_json, connfd) != 0) {
|
||||
rexec_err("send binary information message failed.");
|
||||
goto err_end;
|
||||
}
|
||||
|
||||
pthread_t thrd;
|
||||
pthread_t thrd_conn;
|
||||
struct rexec_thread_arg targ;
|
||||
struct rexec_thread_arg connarg;
|
||||
void *exit_status;
|
||||
targ.efd = pipeefd;
|
||||
(void)pthread_create(&thrd, NULL, rexec_pipe_proxy_thread, &targ);
|
||||
|
||||
connarg.efd = efd;
|
||||
connarg.connfd = connfd;
|
||||
connarg.argv = argv;
|
||||
(void)pthread_create(&thrd_conn, NULL, rexec_conn_thread, &connarg);
|
||||
pthread_join(thrd_conn, (void *)&exit_status);
|
||||
fclose(rexec_logfile);
|
||||
exit((int)exit_status);
|
||||
err_end:
|
||||
fclose(rexec_logfile);
|
||||
rexec_logfile = NULL;
|
||||
return -1;
|
||||
}
|
170
qtfs/rexec/rexec.h
Normal file
170
qtfs/rexec/rexec.h
Normal file
@ -0,0 +1,170 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#ifndef __REXEC_H__
|
||||
#define __REXEC_H__
|
||||
|
||||
#include <time.h>
|
||||
#include <stdbool.h>
|
||||
#include <sys/stat.h>
|
||||
|
||||
enum {
|
||||
PIPE_READ = 0,
|
||||
PIPE_WRITE,
|
||||
};
|
||||
|
||||
enum {
|
||||
REXEC_EVENT_OK,
|
||||
REXEC_EVENT_DEL, // del this event
|
||||
REXEC_EVENT_EXIT, // exit process
|
||||
REXEC_EVENT_ERR,
|
||||
};
|
||||
|
||||
enum {
|
||||
REXEC_STDIN = 0,
|
||||
REXEC_STDOUT,
|
||||
REXEC_STDERR,
|
||||
REXEC_NONE,
|
||||
};
|
||||
|
||||
#define REXEC_MSG_1K 1024
|
||||
#define REXEC_MSG_MAX REXEC_MSG_1K * 1024
|
||||
#define REXEC_LOG_PATH_LEN 64
|
||||
|
||||
// rexec client与server之间建联的sock文件路径
|
||||
#define REXEC_UDS_CONN "/var/run/rexec/rexec_uds.sock"
|
||||
#define REXEC_LOCK_PATH "/var/run/rexec/rexec_uds.lock"
|
||||
#define REXEC_RUN_PATH "/var/run/rexec/"
|
||||
|
||||
enum rexec_msgtype {
|
||||
REXEC_EXEC = 0x5a5a, // exec process
|
||||
REXEC_KILL, // kill process
|
||||
REXEC_PIPE, // client send a pipefd as stdin/out/err to server
|
||||
REXEC_PIDMAP, // server send remote process's pid to client
|
||||
REXEC_HANDSHAKE,
|
||||
};
|
||||
|
||||
struct rexec_msg {
|
||||
int msgtype;
|
||||
// client to server
|
||||
int argc;
|
||||
int pipefd;
|
||||
int msglen;
|
||||
// server to client
|
||||
int exit_status;
|
||||
int pid; // for pidmap
|
||||
char msg[0];
|
||||
};
|
||||
|
||||
extern FILE *rexec_logfile;
|
||||
static inline void rexec_log_init()
|
||||
{
|
||||
char deflog[REXEC_LOG_PATH_LEN] = "/dev/null";
|
||||
char *logfile = getenv("REXEC_LOG_FILE");
|
||||
if (logfile == NULL) {
|
||||
logfile = deflog;
|
||||
} else if (strcmp(logfile, "std") == 0) {
|
||||
rexec_logfile = stderr;
|
||||
return;
|
||||
} else {
|
||||
printf("REXEC_LOG_FILE cannot be set to any value other than std\n");
|
||||
}
|
||||
retry:
|
||||
rexec_logfile = fopen(logfile, "a");
|
||||
if (rexec_logfile == NULL) {
|
||||
if (strcmp(logfile, "/dev/null") == 0) {
|
||||
return;
|
||||
}
|
||||
// 输入的文件打开失败则回退到无日志模式
|
||||
logfile = deflog;
|
||||
goto retry;
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// flag: 1--nonblock; 0--block
|
||||
static inline int rexec_set_nonblock(int fd, int block)
|
||||
{
|
||||
int fflags;
|
||||
if ((fflags = fcntl(fd, F_GETFL)) < 0)
|
||||
return -1;
|
||||
if (block == 0)
|
||||
fflags &= ~O_NONBLOCK;
|
||||
else
|
||||
fflags |= O_NONBLOCK;
|
||||
if ((fcntl(fd, F_SETFL, fflags)) < 0)
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline int rexec_set_inherit(int fd, bool inherit)
|
||||
{
|
||||
int fflags;
|
||||
if ((fflags = fcntl(fd, F_GETFD)) < 0)
|
||||
return -1;
|
||||
if (inherit)
|
||||
fflags &= ~FD_CLOEXEC;
|
||||
else
|
||||
fflags |= FD_CLOEXEC;
|
||||
if ((fcntl(fd, F_SETFD, fflags)) < 0)
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define rexec_log(info, ...) \
|
||||
if (rexec_logfile != NULL) {\
|
||||
time_t t; \
|
||||
struct tm p; \
|
||||
time(&t); \
|
||||
localtime_r(&t, &p); \
|
||||
fprintf(rexec_logfile, "[%d/%02d/%02d %02d:%02d:%02d][LOG:%s:%3d]"info"\n", \
|
||||
p.tm_year + 1900, p.tm_mon+1, p.tm_mday, \
|
||||
p.tm_hour, p.tm_min, p.tm_sec, __func__, __LINE__, ##__VA_ARGS__); \
|
||||
}
|
||||
|
||||
#define rexec_log2(info, ...) \
|
||||
if (rexec_logfile != NULL) {\
|
||||
time_t t; \
|
||||
struct tm p; \
|
||||
time(&t); \
|
||||
localtime_r(&t, &p); \
|
||||
fprintf(rexec_logfile, "[%d/%02d/%02d %02d:%02d:%02d][LOG:%s:%3d]"info"\n", \
|
||||
p.tm_year + 1900, p.tm_mon+1, p.tm_mday, \
|
||||
p.tm_hour, p.tm_min, p.tm_sec, __func__, __LINE__, ##__VA_ARGS__); \
|
||||
}
|
||||
|
||||
#define rexec_err(info, ...) \
|
||||
if (rexec_logfile != NULL) {\
|
||||
time_t t; \
|
||||
struct tm p; \
|
||||
time(&t); \
|
||||
localtime_r(&t, &p); \
|
||||
fprintf(rexec_logfile, "[%d/%02d/%02d %02d:%02d:%02d][ERROR:%s:%3d]"info"\n", \
|
||||
p.tm_year + 1900, p.tm_mon+1, p.tm_mday, \
|
||||
p.tm_hour, p.tm_min, p.tm_sec, __func__, __LINE__, ##__VA_ARGS__); \
|
||||
}
|
||||
|
||||
static inline unsigned int rexec_fd_mode(int fd)
|
||||
{
|
||||
struct stat st;
|
||||
if (fstat(fd, &st) != 0) {
|
||||
rexec_err("get fd:%d fstat failed, errno:%d", fd, errno);
|
||||
return 0;
|
||||
}
|
||||
return st.st_mode;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
615
qtfs/rexec/rexec_server.c
Normal file
615
qtfs/rexec/rexec_server.c
Normal file
@ -0,0 +1,615 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
#include <unistd.h>
|
||||
#include <signal.h>
|
||||
#include <sys/epoll.h>
|
||||
#include <netinet/ip.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/un.h>
|
||||
#include <netinet/udp.h>
|
||||
#include <netinet/tcp.h>
|
||||
#include <sys/socket.h>
|
||||
#include <arpa/inet.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/wait.h>
|
||||
#include <stdbool.h>
|
||||
#include <glib.h>
|
||||
#include <sys/file.h>
|
||||
|
||||
#include "rexec_sock.h"
|
||||
#include "rexec.h"
|
||||
|
||||
static int main_epoll_fd = -1;
|
||||
FILE *rexec_logfile = NULL;
|
||||
static GHashTable *child_hash = NULL;
|
||||
static volatile sig_atomic_t sig_chld_flag = 0;
|
||||
|
||||
#define REXEC_WHITELIST_MAX_ITEMS 256
|
||||
struct rexec_white_list_str {
|
||||
int wl_nums;
|
||||
char *wl[REXEC_WHITELIST_MAX_ITEMS];
|
||||
};
|
||||
static struct rexec_white_list_str rexec_wl;
|
||||
|
||||
extern int rexec_shim_entry(int argc, char *argv[]);
|
||||
|
||||
int rexec_hash_insert_direct(GHashTable *table, int key, int value);
|
||||
int rexec_hash_lookup_direct(GHashTable *table, int key);
|
||||
void rexec_hash_remove_direct(GHashTable *table, int key);
|
||||
|
||||
struct rexec_event {
|
||||
int fd;
|
||||
union {
|
||||
int pid;
|
||||
int connfd;
|
||||
};
|
||||
int (*handler)(struct rexec_event *);
|
||||
};
|
||||
|
||||
static int rexec_add_event(int efd, int fd, int pid, int (*handler)(struct rexec_event *))
|
||||
{
|
||||
struct rexec_event *event = (struct rexec_event *)malloc(sizeof(struct rexec_event));
|
||||
if (event == NULL) {
|
||||
rexec_err("malloc failed.");
|
||||
return -1;
|
||||
}
|
||||
event->fd = fd;
|
||||
event->pid = pid;
|
||||
event->handler = handler;
|
||||
struct epoll_event evt;
|
||||
evt.data.ptr = (void *)event;
|
||||
evt.events = EPOLLIN;
|
||||
if (-1 == epoll_ctl(efd, EPOLL_CTL_ADD, event->fd, &evt)) {
|
||||
rexec_err("epoll ctl add fd:%d event failed.", event->fd);
|
||||
free(event);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rexec_del_event(int efd, struct rexec_event *event)
|
||||
{
|
||||
int ret = epoll_ctl(efd, EPOLL_CTL_DEL, event->fd, NULL);
|
||||
if (ret != 0) {
|
||||
rexec_err("failed to delete event fd:%d.", event->fd);
|
||||
} else {
|
||||
rexec_log("event fd:%d deleted.", event->fd);
|
||||
}
|
||||
close(event->fd);
|
||||
free(event);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rexec_event_process_manage(struct rexec_event *event)
|
||||
{
|
||||
struct rexec_msg head;
|
||||
int ret = recv(event->fd, &head, sizeof(struct rexec_msg), MSG_WAITALL);
|
||||
if (ret <= 0) {
|
||||
rexec_log("Event fd:%d recv ret:%d errno:%d, peer rexec closed, kill the associated process:%d.",
|
||||
event->fd, ret, errno, event->pid);
|
||||
kill(event->pid, SIGKILL);
|
||||
return REXEC_EVENT_DEL;
|
||||
}
|
||||
rexec_err("Recv msg from client, msgtype:%d msglen:%d argc:%d pipefd:%d",
|
||||
head.msgtype, head.msglen, head.argc, head.pipefd);
|
||||
return REXEC_EVENT_OK;
|
||||
}
|
||||
|
||||
static int rexec_event_handshake(struct rexec_event *event)
|
||||
{
|
||||
int sonpid = 0;
|
||||
int ret = read(event->fd, &sonpid, sizeof(int));
|
||||
if (ret <= 0) {
|
||||
rexec_err("Rexec read from pipe ret:%d errno:%d", ret, errno);
|
||||
return REXEC_EVENT_DEL;
|
||||
}
|
||||
int connfd = event->connfd;
|
||||
if (sonpid == -1) {
|
||||
rexec_err("Handshake recv -1, dont add to process manage");
|
||||
close(connfd);
|
||||
return REXEC_EVENT_DEL;
|
||||
}
|
||||
rexec_log("Rexec recv son pid:%d, connfd:%d", sonpid, connfd);
|
||||
|
||||
rexec_hash_insert_direct(child_hash, sonpid, connfd);
|
||||
rexec_add_event(main_epoll_fd, connfd, sonpid, rexec_event_process_manage);
|
||||
|
||||
// 成功后同样要删除这个pipe监听事件,删除时会close掉fd
|
||||
return REXEC_EVENT_DEL;
|
||||
}
|
||||
|
||||
static int rexec_dup_pipefd(int fd, int pipefd)
|
||||
{
|
||||
int dupfd;
|
||||
if (fd == -1)
|
||||
return 0;
|
||||
dupfd = dup2(fd, pipefd);
|
||||
if (dupfd != pipefd) {
|
||||
rexec_err("failed to dup pipefd:%d", pipefd);
|
||||
return -1;
|
||||
}
|
||||
close(fd);
|
||||
return 0;
|
||||
}
|
||||
|
||||
// argv list: [0]binary,[1]-f,[2]*json_str,[3]arg1,[4]arg2,...
|
||||
static int rexec_parse_argv(int argc, char *argv_str, char *argv[])
|
||||
{
|
||||
int offset = 0;
|
||||
for (int i = 0; i < argc; i++) {
|
||||
argv[i] = &argv_str[offset];
|
||||
offset += strlen(argv[i]) + 1;
|
||||
}
|
||||
argv[argc] = NULL;
|
||||
return offset;
|
||||
}
|
||||
|
||||
static inline void rexec_clear_string_tail(char *str, int len)
|
||||
{
|
||||
while (len >= 0 && str[len] < 0x20) {
|
||||
str[len] = '\0';
|
||||
len--;
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
#define REXEC_WHITELIST_FILE "/etc/rexec/whitelist"
|
||||
static int rexec_whitelist_build(struct rexec_white_list_str *wl)
|
||||
{
|
||||
if (access(REXEC_WHITELIST_FILE, F_OK) != 0) {
|
||||
rexec_err("Please configure the white list(%s).", REXEC_WHITELIST_FILE);
|
||||
return -1;
|
||||
}
|
||||
|
||||
wl->wl_nums = 0;
|
||||
memset(wl->wl, 0, sizeof(uintptr_t) * REXEC_WHITELIST_MAX_ITEMS);
|
||||
#define MAX_CMD_LEN 256
|
||||
char cmd[MAX_CMD_LEN];
|
||||
FILE *fwl = fopen(REXEC_WHITELIST_FILE, "r");
|
||||
if (fwl == NULL) {
|
||||
rexec_err("open white list file:%s failed.", REXEC_WHITELIST_FILE);
|
||||
return -1;
|
||||
}
|
||||
struct stat stats;
|
||||
int ret = fstat(fileno(fwl), &stats);
|
||||
if (ret != 0) {
|
||||
rexec_err("fstat white list file:%s failed.", REXEC_WHITELIST_FILE);
|
||||
goto err_end;
|
||||
}
|
||||
if (stats.st_mode & 0777 != 0400) {
|
||||
rexec_err("white list file:%s permissions(%o) error, must be read-only(0400)", stats.st_mode, REXEC_WHITELIST_FILE);
|
||||
goto err_end;
|
||||
}
|
||||
while (!feof(fwl) && wl->wl_nums < REXEC_WHITELIST_MAX_ITEMS) {
|
||||
int len;
|
||||
char *fstr;
|
||||
memset(cmd, 0, MAX_CMD_LEN);
|
||||
fstr = fgets(cmd, MAX_CMD_LEN - 1, fwl);
|
||||
if (fstr == NULL)
|
||||
continue;
|
||||
rexec_clear_string_tail(cmd, strlen(cmd));
|
||||
len = strlen(cmd);
|
||||
fstr = (char *)malloc(len + 1);
|
||||
if (fstr == NULL) {
|
||||
rexec_err("Malloc failed");
|
||||
goto err_end;
|
||||
}
|
||||
memset(fstr, 0, len + 1);
|
||||
memcpy(fstr, cmd, len);
|
||||
wl->wl[wl->wl_nums] = fstr;
|
||||
wl->wl_nums++;
|
||||
rexec_log("Cmd:<%s> added to white list.", cmd);
|
||||
}
|
||||
fclose(fwl);
|
||||
if (wl->wl_nums == 0) {
|
||||
rexec_err("White list file [%s] has no valid content.", REXEC_WHITELIST_FILE);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
|
||||
err_end:
|
||||
for (int i = 0; i < wl->wl_nums; i++) {
|
||||
free(wl->wl[i]);
|
||||
}
|
||||
fclose(fwl);
|
||||
return -1;
|
||||
}
|
||||
|
||||
static void rexec_white_list_free(struct rexec_white_list_str *wl)
|
||||
{
|
||||
for (int i = 0; i < wl->wl_nums; i++) {
|
||||
free(wl->wl[i]);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
static int rexec_whitelist_check(char *binary)
|
||||
{
|
||||
// white list file not exist, always ok
|
||||
if (access(REXEC_WHITELIST_FILE, F_OK) != 0)
|
||||
return -1;
|
||||
for (int i = 0; i < rexec_wl.wl_nums; i++) {
|
||||
if (strlen(binary) == strlen(rexec_wl.wl[i]) && strcmp(binary, rexec_wl.wl[i]) == 0)
|
||||
return 0;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
#define IS_VALID_FD(fd) (fd > STDERR_FILENO)
|
||||
static void handle_sig_chld(void)
|
||||
{
|
||||
int status;
|
||||
pid_t pid;
|
||||
while ((pid = waitpid(0, &status, WNOHANG)) > 0) {
|
||||
int exit_status = status;
|
||||
if (WIFEXITED(status)) {
|
||||
exit_status = WEXITSTATUS(status);
|
||||
} else if (WIFSIGNALED(status)) {
|
||||
exit_status = WTERMSIG(status) + 128;
|
||||
}
|
||||
int connfd = rexec_hash_lookup_direct(child_hash, pid);
|
||||
if (IS_VALID_FD(connfd)) {
|
||||
struct rexec_msg head;
|
||||
head.msgtype = REXEC_KILL;
|
||||
head.msglen = 0;
|
||||
head.exit_status = exit_status;
|
||||
rexec_sendmsg(connfd, (char *)&head, sizeof(struct rexec_msg), -1);
|
||||
rexec_hash_remove_direct(child_hash, pid);
|
||||
// don't close connfd
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
static void rexec_server_sig_chld(int num)
|
||||
{
|
||||
__sync_fetch_and_add(&sig_chld_flag, 1);
|
||||
return;
|
||||
}
|
||||
|
||||
static void rexec_server_sig_pipe(int signum)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
#define REXEC_MSG_NORMAL (1 << 3)
|
||||
#define REXEC_MSG_OVER 0xf
|
||||
static int rexec_start_new_process(int newconnfd)
|
||||
{
|
||||
int ret;
|
||||
int pipefd[2];
|
||||
if (pipe(pipefd) == -1) {
|
||||
rexec_err("pipe syscall error, errno:%d", errno);
|
||||
return -1;
|
||||
}
|
||||
// handshake阶段,联合体里面记录newconnfd
|
||||
// 等到handshake成功后,新的事件监听这个newconnfd,联合体改为记录son pid
|
||||
rexec_add_event(main_epoll_fd, pipefd[PIPE_READ], newconnfd, rexec_event_handshake);
|
||||
|
||||
int pid = fork();
|
||||
// parent
|
||||
if (pid != 0) {
|
||||
close(pipefd[PIPE_WRITE]);
|
||||
return 0;
|
||||
}
|
||||
// son
|
||||
close(pipefd[PIPE_READ]);
|
||||
|
||||
struct rexec_msg head;
|
||||
int argc;
|
||||
char *msgbuf = NULL;
|
||||
char msg_bit = 0;
|
||||
while (msg_bit != REXEC_MSG_OVER) {
|
||||
int scmfd = -1;
|
||||
int len = sizeof(struct rexec_msg);
|
||||
memset(&head, 0, sizeof(struct rexec_msg));
|
||||
scmfd = -1;
|
||||
ret = rexec_recvmsg(newconnfd, (char *)&head, len, &scmfd, MSG_WAITALL);
|
||||
if (ret <= 0) {
|
||||
rexec_log("recvmsg ret:%d, errno:%d", ret, errno);
|
||||
goto err_to_parent;
|
||||
}
|
||||
// 将管道与自己的标准输入输出关联
|
||||
if (rexec_dup_pipefd(scmfd, head.pipefd) != 0) {
|
||||
rexec_err("dup scm:%d pipefd:%d failed", scmfd, head.pipefd);
|
||||
goto err_to_parent;
|
||||
}
|
||||
if (head.pipefd >= REXEC_STDIN && head.pipefd <= REXEC_STDERR) {
|
||||
msg_bit |= (1 << (head.pipefd - REXEC_STDIN));
|
||||
}
|
||||
if (head.msglen == 0)
|
||||
continue;
|
||||
// 普通消息,代码暂时没考虑多个普通消息的,先直接过滤
|
||||
if (msgbuf != NULL || head.msgtype != REXEC_EXEC) {
|
||||
rexec_err("not support multi normal msg or msgtype:%d msglen:%d invalid", head.msgtype, head.msglen);
|
||||
continue;
|
||||
}
|
||||
msg_bit |= REXEC_MSG_NORMAL;
|
||||
// exec msg
|
||||
rexec_log("Exec msgtype:0x%x msglen:%d argc:%d stdno:%d",
|
||||
head.msgtype, head.msglen, head.argc, head.pipefd);
|
||||
argc = head.argc;
|
||||
if (head.msglen > REXEC_MSG_MAX || argc > REXEC_MSG_MAX / sizeof(uintptr_t) ||
|
||||
head.msglen <= 0 || argc < 0) {
|
||||
rexec_err("msg len:%d or argc:%d is too large", head.msglen, argc);
|
||||
goto err_to_parent;
|
||||
}
|
||||
msgbuf = (char *)malloc(head.msglen + 1);
|
||||
if (msgbuf == NULL) {
|
||||
rexec_err("malloc failed");
|
||||
goto err_to_parent;
|
||||
}
|
||||
memset(msgbuf, 0, head.msglen + 1);
|
||||
ret = recv(newconnfd, msgbuf, head.msglen, MSG_WAITALL);
|
||||
if (ret <= 0) {
|
||||
rexec_err("recv failed, ret:%d errno:%d", ret, errno);
|
||||
goto err_free;
|
||||
}
|
||||
rexec_log("recv normal msg len:%d headlen:%d real recv:%d msg:%s",
|
||||
head.msglen, sizeof(struct rexec_msg), ret, msgbuf);
|
||||
}
|
||||
|
||||
// msg is like: "binary"\0"argv[1]"\0"argv[2]"\0"..."
|
||||
char *binary = msgbuf;
|
||||
if (rexec_whitelist_check(binary) != 0) {
|
||||
rexec_err("Cmd:<%s> not in white list.", binary);
|
||||
goto err_free;
|
||||
}
|
||||
|
||||
char *ack;
|
||||
int mypid = getpid();
|
||||
char msg[sizeof(struct rexec_msg) + 1];
|
||||
struct rexec_msg *pm = msg;
|
||||
pm->msgtype = REXEC_PIDMAP;
|
||||
pm->msglen = 0;
|
||||
pm->pid = mypid;
|
||||
ret = write(newconnfd, pm, sizeof(struct rexec_msg));
|
||||
if (ret <= 0) {
|
||||
rexec_err("Rexec send son pid:%d to client failed, ret:%d errno:%d", mypid, ret, errno);
|
||||
} else {
|
||||
retry:
|
||||
rexec_log("Waiting for rexec client handshake...");
|
||||
ret = read(newconnfd, pm, sizeof(struct rexec_msg) + 1);
|
||||
if (ret <= 0) {
|
||||
rexec_err("Recv handshake failed, ret:%d err:%d", ret, errno);
|
||||
goto err_to_parent;
|
||||
}
|
||||
if (pm->msgtype != REXEC_HANDSHAKE) {
|
||||
rexec_err("Recv unexpected msg:%d", pm->msgtype);
|
||||
goto retry;
|
||||
}
|
||||
ack = pm->msg;
|
||||
if (*ack != '1') {
|
||||
rexec_err("recv error handshake ack from client:%c, exit now", *ack);
|
||||
goto err_to_parent;
|
||||
}
|
||||
}
|
||||
// 写会PID必须放在基于newconnfd接收完所有消息之后,
|
||||
// 后面newconnfd的控制权交回父进程rexec server服务进程
|
||||
if (write(pipefd[PIPE_WRITE], &mypid, sizeof(int)) <= 0) {
|
||||
rexec_err("write pid to parent failed, pipefd:%d.", pipefd[PIPE_WRITE]);
|
||||
}
|
||||
// 子进程不再使用pipe write和connfd
|
||||
close(pipefd[PIPE_WRITE]);
|
||||
close(newconnfd);
|
||||
|
||||
rexec_log("handshake over normaly, continue to exec new process:%s.", binary);
|
||||
|
||||
// rexec_shim_entry argv like:
|
||||
// argv[0]: binary
|
||||
// argv[1]: -f
|
||||
// argv[2]: *json_str
|
||||
// argv[3]: param list 1
|
||||
// argv[4]: ...
|
||||
char **argv = (char **)malloc(sizeof(uintptr_t) * (argc + 3));
|
||||
if (argv == NULL) {
|
||||
rexec_err("malloc failed, argc:%d.", argc);
|
||||
goto err_free;
|
||||
}
|
||||
int offset = rexec_parse_argv(argc, msgbuf, &argv[2]);
|
||||
argv[0] = "-f";
|
||||
argv[1] = &msgbuf[offset];
|
||||
|
||||
rexec_log("Parse argv result argc:%d", argc);
|
||||
for (int i = 2; i < argc + 2; i++) {
|
||||
rexec_log(" argv[%d]:%s", i - 2, argv[i]);
|
||||
}
|
||||
ret = rexec_shim_entry(argc + 2, argv);
|
||||
perror("rexec shim entry");
|
||||
|
||||
err_free:
|
||||
free(msgbuf);
|
||||
|
||||
err_to_parent:
|
||||
do {
|
||||
int errpid = -1;
|
||||
write(pipefd[PIPE_WRITE], &errpid, sizeof(int));
|
||||
} while (0);
|
||||
|
||||
exit(0);
|
||||
}
|
||||
|
||||
// 道生一
|
||||
static int rexec_event_new_process(struct rexec_event *event)
|
||||
{
|
||||
int newconnfd = rexec_sock_step_accept(event->fd);
|
||||
if (newconnfd < 0) {
|
||||
rexec_err("Accept failed, ret:%d errno:%d", newconnfd, errno);
|
||||
return REXEC_EVENT_ERR;
|
||||
}
|
||||
// 主进程只负责接收新链接,基于newconnfd的新消息由子进程自己去接收,但是最
|
||||
// 后父进程要进入监听此链接的状态,是为了联动kill(对端杀死client进程则本端
|
||||
// 也杀死,或者本端杀死子进程后消息通知对端也杀死)
|
||||
// 这个fd不能同时被父子进程监听,所以先建立一个pipe,等子进程完全接收完
|
||||
// 初始消息后,通过pipe告知父进程再由父进程接管newconnfd,在这之前,父进程
|
||||
// 监听pipe的read端
|
||||
// 白名单也在子进程里做,在fork之后,rexec代码控制范围
|
||||
rexec_log("Start new process new conn fd:%d", newconnfd);
|
||||
rexec_start_new_process(newconnfd);
|
||||
return REXEC_EVENT_OK;
|
||||
}
|
||||
|
||||
static void rexec_server_mainloop()
|
||||
{
|
||||
#define REXEC_MAX_EVENTS 16
|
||||
main_epoll_fd = epoll_create1(0);
|
||||
if (main_epoll_fd == -1) {
|
||||
rexec_err("epoll create1 failed, errno:%d.", errno);
|
||||
return;
|
||||
}
|
||||
if (rexec_set_inherit(main_epoll_fd, false) < 0) {
|
||||
rexec_err("epoll fd set inherit to false failed.");
|
||||
}
|
||||
struct rexec_conn_arg ser = {
|
||||
.cs = REXEC_SOCK_SERVER,
|
||||
.udstype = SOCK_STREAM,
|
||||
};
|
||||
strncpy(ser.sun_path, REXEC_UDS_CONN, strlen(REXEC_UDS_CONN) + 1);
|
||||
int buildret = rexec_build_unix_connection(&ser);
|
||||
if (buildret != 0) {
|
||||
rexec_err("faild to build main sock:%d errno:%d", buildret, errno);
|
||||
close(main_epoll_fd);
|
||||
return;
|
||||
}
|
||||
if (chmod(REXEC_UDS_CONN, 0600) < 0) {
|
||||
rexec_err("failed to set uds sock file mode:%s errno:%d", REXEC_UDS_CONN, errno);
|
||||
close(main_epoll_fd);
|
||||
close(ser.sockfd);
|
||||
return;
|
||||
}
|
||||
if (rexec_set_inherit(ser.sockfd, false) < 0) {
|
||||
rexec_err("cs conn fd fd set inherit to false failed.");
|
||||
}
|
||||
rexec_add_event(main_epoll_fd, ser.sockfd, 0, rexec_event_new_process);
|
||||
|
||||
struct epoll_event *evts = calloc(REXEC_MAX_EVENTS, sizeof(struct epoll_event));
|
||||
if (evts == NULL) {
|
||||
rexec_err("init calloc evts failed.");
|
||||
goto end;
|
||||
}
|
||||
while (1) {
|
||||
if (sig_chld_flag != 0 && __sync_fetch_and_sub(&sig_chld_flag, 1) != 0) {
|
||||
handle_sig_chld();
|
||||
}
|
||||
int n = epoll_wait(main_epoll_fd, evts, REXEC_MAX_EVENTS, 1000);
|
||||
if (n == 0)
|
||||
continue;
|
||||
if (n < 0) {
|
||||
rexec_err("epoll wait return errcode:%d", n);
|
||||
continue;
|
||||
}
|
||||
rexec_log("epoll wait trigger %d events.", n);
|
||||
for (int i = 0; i < n; i++) {
|
||||
struct rexec_event *event = (struct rexec_event *)evts[i].data.ptr;
|
||||
int ret = event->handler(event);
|
||||
if (ret == REXEC_EVENT_DEL)
|
||||
rexec_del_event(main_epoll_fd, event);
|
||||
}
|
||||
}
|
||||
free(evts);
|
||||
|
||||
end:
|
||||
close(main_epoll_fd);
|
||||
main_epoll_fd = -1;
|
||||
close(ser.sockfd);
|
||||
ser.sockfd = -1;
|
||||
return;
|
||||
}
|
||||
|
||||
// hash map for child pid and conn fd
|
||||
int rexec_pid_hashmap_init(GHashTable **table)
|
||||
{
|
||||
*table = g_hash_table_new(g_direct_hash, g_direct_equal);
|
||||
if (*table == NULL) {
|
||||
rexec_err("Init child pid hashmap failed.");
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
void rexec_pid_hashmap_destroy(GHashTable *table)
|
||||
{
|
||||
g_hash_table_destroy(table);
|
||||
return;
|
||||
}
|
||||
|
||||
#pragma GCC diagnostic ignored "-Wint-to-pointer-cast"
|
||||
#pragma GCC diagnostic ignored "-Wpointer-to-int-cast"
|
||||
int rexec_hash_insert_direct(GHashTable *table, int key, int value)
|
||||
{
|
||||
if (g_hash_table_insert(table, (gpointer)key, (gpointer)value) == 0) {
|
||||
rexec_err("Hash table key:%d value:%d is already exist, update it.", key, value);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int rexec_hash_lookup_direct(GHashTable *table, int key)
|
||||
{
|
||||
return (int)g_hash_table_lookup(table, (gpointer)key);
|
||||
}
|
||||
|
||||
void rexec_hash_remove_direct(GHashTable *table, int key)
|
||||
{
|
||||
g_hash_table_remove(table, (gpointer)key);
|
||||
return;
|
||||
}
|
||||
#pragma GCC diagnostic pop
|
||||
|
||||
int check_socket_lock(void)
|
||||
{
|
||||
int lock_fd = open(REXEC_LOCK_PATH, O_RDONLY | O_CREAT, 0600);
|
||||
if (lock_fd == -1)
|
||||
return -EINVAL;
|
||||
|
||||
return flock(lock_fd, LOCK_EX | LOCK_NB);
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
mode_t newmask = 0077;
|
||||
rexec_log("Change umask from:%o to %o", umask(newmask), newmask);
|
||||
rexec_log_init();
|
||||
signal(SIGCHLD, rexec_server_sig_chld);
|
||||
signal(SIGPIPE, rexec_server_sig_pipe);
|
||||
if (rexec_whitelist_build(&rexec_wl) != 0) {
|
||||
return -1;
|
||||
}
|
||||
if (access(REXEC_RUN_PATH, F_OK) != 0) {
|
||||
mkdir(REXEC_RUN_PATH, 0700);
|
||||
}
|
||||
|
||||
if (check_socket_lock() < 0) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (rexec_pid_hashmap_init(&child_hash) != 0) {
|
||||
rexec_white_list_free(&rexec_wl);
|
||||
return -1;
|
||||
}
|
||||
rexec_server_mainloop();
|
||||
rexec_pid_hashmap_destroy(child_hash);
|
||||
fclose(rexec_logfile);
|
||||
rexec_logfile = NULL;
|
||||
return 0;
|
||||
}
|
||||
|
157
qtfs/rexec/rexec_shim.c
Normal file
157
qtfs/rexec/rexec_shim.c
Normal file
@ -0,0 +1,157 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
#include <errno.h>
|
||||
#include <time.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <json-c/json_object.h>
|
||||
#include <json-c/json_tokener.h>
|
||||
|
||||
#include "dirent.h"
|
||||
#include "rexec.h"
|
||||
|
||||
#define rshim_log rexec_log
|
||||
#define rshim_err rexec_err
|
||||
|
||||
void rshim_close_all_fd()
|
||||
{
|
||||
DIR *dir = NULL;
|
||||
struct dirent *entry;
|
||||
dir = opendir("/proc/self/fd/");
|
||||
if (dir == NULL) {
|
||||
rshim_err("open path:/proc/self/fd/ failed");
|
||||
return;
|
||||
}
|
||||
while (entry = readdir(dir)) {
|
||||
int fd = atoi(entry->d_name);
|
||||
if (fd <= 2 || S_ISFIFO(rexec_fd_mode(fd)))
|
||||
continue;
|
||||
close(fd);
|
||||
}
|
||||
closedir(dir);
|
||||
return;
|
||||
}
|
||||
|
||||
int rshim_get_file_size(char *file)
|
||||
{
|
||||
int size = 0;
|
||||
FILE *f = fopen(file, "rb");
|
||||
if (f == NULL) {
|
||||
rshim_err("File:%s fopen failed.", file);
|
||||
return -1;
|
||||
}
|
||||
fseek(f, 0, SEEK_END);
|
||||
size = ftell(f);
|
||||
fclose(f);
|
||||
return size;
|
||||
}
|
||||
|
||||
void rshim_reg_file_open(int fd_target, const char *path, int perm, int offset)
|
||||
{
|
||||
int fd = open(path, perm);
|
||||
int fd2 = -1;
|
||||
if (fd < 0) {
|
||||
rshim_err("Open file:%s failed, fd:%d errno:%d", path, fd, errno);
|
||||
return;
|
||||
}
|
||||
if (fd != fd_target) {
|
||||
fd2 = dup2(fd, fd_target);
|
||||
if (fd2 != fd_target) {
|
||||
rshim_err("Failed to open file:%s by fd:%d", path, fd_target);
|
||||
close(fd2);
|
||||
close(fd);
|
||||
return;
|
||||
}
|
||||
close(fd);
|
||||
}
|
||||
int off = lseek(fd_target, offset, SEEK_SET);
|
||||
if (off < 0) {
|
||||
rshim_err("Failed to set offset:%d to file:%s, fd:%d, fd2:%d", offset, path, fd, fd2);
|
||||
return;
|
||||
}
|
||||
rshim_log("Successed to set offset:%d to file:%s, fd:%d, fd2:%d", offset, path, fd, fd2);
|
||||
return;
|
||||
}
|
||||
|
||||
void rshim_reg_file_resume(const char * const json_buf)
|
||||
{
|
||||
struct json_object *obj_files;
|
||||
struct json_object *obj_file;
|
||||
struct json_object *obj_fd;
|
||||
struct json_object *obj_path;
|
||||
struct json_object *obj_perm;
|
||||
struct json_object *obj_offset;
|
||||
int fd, perm, offset;
|
||||
const char *path = NULL;
|
||||
int curfd = 3; // begin from 3
|
||||
struct json_object *fd_json = json_tokener_parse(json_buf);
|
||||
if (fd_json == NULL) {
|
||||
fprintf(stderr, "parse json error\n");
|
||||
return;
|
||||
}
|
||||
obj_files = json_object_object_get(fd_json, "Files");
|
||||
int arraylen = json_object_array_length(obj_files);
|
||||
for (int i=0; i< arraylen; i++){
|
||||
obj_file = json_object_array_get_idx(obj_files, i);
|
||||
obj_fd = json_object_object_get(obj_file, "Fd");
|
||||
fd = json_object_get_int(obj_fd);
|
||||
obj_path = json_object_object_get(obj_file, "Path");
|
||||
path = json_object_get_string(obj_path);
|
||||
obj_perm = json_object_object_get(obj_file, "Perm");
|
||||
perm = json_object_get_int(obj_perm);
|
||||
obj_offset = json_object_object_get(obj_file, "Offset");
|
||||
offset = json_object_get_int(obj_offset);
|
||||
rshim_log("Get file from json fd:%d path:%s perm:%d offset:%d",
|
||||
fd, path, perm, offset);
|
||||
rshim_reg_file_open(fd, path, perm, offset);
|
||||
}
|
||||
|
||||
json_object_put(fd_json);
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
param list:
|
||||
1) -f xxx.json binary param1 param2 ...
|
||||
2) binary param1 param2...
|
||||
*/
|
||||
int rexec_shim_entry(int argc, char *argv[])
|
||||
{
|
||||
char *json_str = NULL;
|
||||
char **newarg = NULL;
|
||||
|
||||
if (strcmp(argv[0], "-f") == 0) {
|
||||
json_str = argv[1];
|
||||
newarg = &argv[2];
|
||||
} else {
|
||||
newarg = argv;
|
||||
}
|
||||
|
||||
rshim_close_all_fd();
|
||||
|
||||
rshim_log("Get json str:%s", json_str);
|
||||
|
||||
rshim_reg_file_resume(json_str);
|
||||
execvp(newarg[0], newarg);
|
||||
perror("execvp failed.");
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
177
qtfs/rexec/rexec_sock.c
Normal file
177
qtfs/rexec/rexec_sock.c
Normal file
@ -0,0 +1,177 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
#include <unistd.h>
|
||||
#include <signal.h>
|
||||
#include <sys/epoll.h>
|
||||
#include <netinet/ip.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/un.h>
|
||||
#include <netinet/udp.h>
|
||||
#include <netinet/tcp.h>
|
||||
#include <sys/socket.h>
|
||||
#include <arpa/inet.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
|
||||
#include "rexec.h"
|
||||
#include "rexec_sock.h"
|
||||
|
||||
int rexec_build_unix_connection(struct rexec_conn_arg *arg)
|
||||
{
|
||||
const int sock_max_conn = 5;
|
||||
if (arg->cs > REXEC_SOCK_SERVER) {
|
||||
rexec_err("cs type %d is error.", arg->cs);
|
||||
return -1;
|
||||
}
|
||||
struct sockaddr_un sock_addr = {
|
||||
.sun_family = AF_UNIX,
|
||||
};
|
||||
int sock_fd = socket(AF_UNIX, arg->udstype, 0);
|
||||
|
||||
if (sock_fd < 0) {
|
||||
rexec_err("As %s failed, socket fd: %d, errno:%d.",
|
||||
(arg->cs == REXEC_SOCK_CLIENT) ? "client" : "server",
|
||||
sock_fd, errno);
|
||||
return -1;
|
||||
}
|
||||
strncpy(sock_addr.sun_path, arg->sun_path, sizeof(sock_addr.sun_path));
|
||||
arg->sockfd = sock_fd;
|
||||
|
||||
if (arg->cs == REXEC_SOCK_SERVER) {
|
||||
unlink(sock_addr.sun_path);
|
||||
if (bind(sock_fd, (struct sockaddr *)&sock_addr, sizeof(sock_addr)) < 0) {
|
||||
rexec_err("As server failed, bind error, errno:%d.",
|
||||
errno);
|
||||
goto close_and_return;
|
||||
}
|
||||
if (listen(sock_fd, sock_max_conn) < 0) {
|
||||
rexec_err("As server listen failed, errno:%d.", errno);
|
||||
goto close_and_return;
|
||||
}
|
||||
} else {
|
||||
if (connect(arg->sockfd, (struct sockaddr *)&sock_addr, sizeof(struct sockaddr_un)) < 0) {
|
||||
goto close_and_return;
|
||||
}
|
||||
arg->connfd = sock_fd;
|
||||
rexec_log("Connect to server successed, sun path:%s", arg->sun_path);
|
||||
}
|
||||
|
||||
return 0;
|
||||
close_and_return:
|
||||
rexec_log("close sockfd:%d and return", sock_fd);
|
||||
close(sock_fd);
|
||||
return -1;
|
||||
|
||||
}
|
||||
|
||||
int rexec_sock_step_accept(int sock_fd)
|
||||
{
|
||||
struct sockaddr_un un_addr;
|
||||
socklen_t len = sizeof(struct sockaddr_un);
|
||||
int connfd;
|
||||
connfd = accept(sock_fd, (struct sockaddr *)&un_addr, &len);
|
||||
if (connfd < 0) {
|
||||
rexec_err("Accept error:%d, errno:%d.", connfd, errno);
|
||||
return connfd;
|
||||
}
|
||||
rexec_log("Accept success, sun path:%s", un_addr.sun_path);
|
||||
return connfd;
|
||||
}
|
||||
|
||||
// send a normal msg
|
||||
// or a SCM_RIGHTS fd
|
||||
// or a normal msg and a SCM_RIGHTS fd
|
||||
int rexec_sendmsg(int sockfd, char *msgbuf, int msglen, int scmfd)
|
||||
{
|
||||
struct msghdr msg;
|
||||
struct cmsghdr *cmsg = NULL;
|
||||
struct iovec iov;
|
||||
char buf[CMSG_SPACE(sizeof(scmfd))];
|
||||
int ret;
|
||||
|
||||
memset(&msg, 0, sizeof(msg));
|
||||
iov.iov_base = (void *)msgbuf;
|
||||
iov.iov_len = msglen;
|
||||
msg.msg_iov = &iov;
|
||||
msg.msg_iovlen = 1;
|
||||
msg.msg_name = NULL;
|
||||
msg.msg_namelen = 0;
|
||||
|
||||
msg.msg_control = buf;
|
||||
msg.msg_controllen = sizeof(buf);
|
||||
if (scmfd > 0) {
|
||||
cmsg = CMSG_FIRSTHDR(&msg);
|
||||
cmsg->cmsg_level = SOL_SOCKET;
|
||||
cmsg->cmsg_type = SCM_RIGHTS;
|
||||
cmsg->cmsg_len = CMSG_LEN(sizeof(scmfd));
|
||||
/* Initialize the payload: */
|
||||
memcpy(CMSG_DATA(cmsg), &scmfd, sizeof(scmfd));
|
||||
msg.msg_controllen = cmsg->cmsg_len;
|
||||
} else {
|
||||
msg.msg_controllen = 0;
|
||||
}
|
||||
if ((ret = sendmsg(sockfd, &msg, 0)) != iov.iov_len) {
|
||||
return ret;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int rexec_recvmsg(int sockfd, char *msgbuf, int len, int *scmfd, int flags)
|
||||
{
|
||||
struct iovec iov;
|
||||
struct msghdr msg;
|
||||
int fd = -1;
|
||||
struct cmsghdr *cmsg;
|
||||
char buf[CMSG_SPACE(sizeof(fd))];
|
||||
|
||||
/* send at least one char */
|
||||
memset(&msg, 0, sizeof(msg));
|
||||
iov.iov_base = msgbuf;
|
||||
iov.iov_len = len;
|
||||
msg.msg_iov = &iov;
|
||||
msg.msg_iovlen = 1;
|
||||
msg.msg_name = NULL;
|
||||
msg.msg_namelen = 0;
|
||||
|
||||
msg.msg_control = buf;
|
||||
msg.msg_controllen = sizeof(buf);
|
||||
cmsg = CMSG_FIRSTHDR(&msg);
|
||||
cmsg->cmsg_level = SOL_SOCKET;
|
||||
cmsg->cmsg_type = SCM_RIGHTS;
|
||||
cmsg->cmsg_len = CMSG_LEN(sizeof(fd));
|
||||
/* Initialize the payload: */
|
||||
memcpy(CMSG_DATA(cmsg), &fd, sizeof(fd));
|
||||
msg.msg_controllen = cmsg->cmsg_len;
|
||||
|
||||
int ret = recvmsg(sockfd, &msg, flags);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
cmsg = CMSG_FIRSTHDR(&msg);
|
||||
if (cmsg != NULL) {
|
||||
memcpy(&fd, CMSG_DATA(cmsg), sizeof(fd));
|
||||
*scmfd = fd;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
40
qtfs/rexec/rexec_sock.h
Normal file
40
qtfs/rexec/rexec_sock.h
Normal file
@ -0,0 +1,40 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#ifndef __REXEC_SOCK_H__
|
||||
#define __REXEC_SOCK_H__
|
||||
|
||||
enum {
|
||||
REXEC_SOCK_CLIENT = 1,
|
||||
REXEC_SOCK_SERVER,
|
||||
};
|
||||
|
||||
#define UDS_SUN_PATH_LEN 108
|
||||
struct rexec_conn_arg {
|
||||
int cs; // client(1) or server(2)
|
||||
|
||||
int udstype; // DGRAM or STREAM
|
||||
char sun_path[UDS_SUN_PATH_LEN];
|
||||
int sockfd;
|
||||
int connfd;
|
||||
};
|
||||
|
||||
int rexec_sock_step_accept(int sock_fd);
|
||||
int rexec_build_unix_connection(struct rexec_conn_arg *arg);
|
||||
int rexec_sendmsg(int sockfd, char *msgbuf, int msglen, int scmfd);
|
||||
int rexec_recvmsg(int sockfd, char *msgbuf, int msglen, int *scmfd, int flags);
|
||||
|
||||
#endif
|
||||
|
53
qtfs/test/cgroup.go
Normal file
53
qtfs/test/cgroup.go
Normal file
@ -0,0 +1,53 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"path"
|
||||
"sync"
|
||||
"time"
|
||||
"strconv"
|
||||
)
|
||||
|
||||
func do_watch(dir, prefix string) {
|
||||
entries, err := ioutil.ReadDir(dir)
|
||||
if err != nil {
|
||||
fmt.Printf("read cgroup dir(%s) failed: %s\n", dir, err.Error())
|
||||
return
|
||||
}
|
||||
|
||||
for _, entry := range entries {
|
||||
if entry.IsDir() {
|
||||
entryPath := path.Join(dir, entry.Name())
|
||||
prefix = prefix + " "
|
||||
do_watch(entryPath, prefix)
|
||||
} else {
|
||||
filePath := path.Join(dir, entry.Name())
|
||||
file, err := os.Open(filePath)
|
||||
if err == nil {
|
||||
file.Close()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func watch(dir, prefix string, wg *sync.WaitGroup) {
|
||||
do_watch(dir, prefix)
|
||||
wg.Done()
|
||||
}
|
||||
|
||||
func main() {
|
||||
var wg sync.WaitGroup
|
||||
begin := time.Now()
|
||||
threads, _ := strconv.Atoi(os.Args[2])
|
||||
fmt.Printf("watch %s\n", os.Args[1])
|
||||
for i := 0; i < threads; i++ {
|
||||
wg.Add(1)
|
||||
go watch(os.Args[1], "", &wg)
|
||||
fmt.Printf("Thread run %d\n", i)
|
||||
}
|
||||
wg.Wait()
|
||||
dlt := time.Since(begin)
|
||||
fmt.Printf("All thread over, %d threads cost time:%v\n", threads, dlt)
|
||||
}
|
30
qtfs/test/pjdfstest/README.md
Normal file
30
qtfs/test/pjdfstest/README.md
Normal file
@ -0,0 +1,30 @@
|
||||
# 开源文件系统测试集pjdfstest
|
||||
|
||||
pjdfstest是一个POSIX系统接口的测试套,用于进行文件系统接口兼容性测试,相关说明及源码可见其开源链接:
|
||||
|
||||
[pjdfstest](https://github.com/pjd/pjdfstest)
|
||||
|
||||
### 部署
|
||||
|
||||
```bash
|
||||
# qtfs仍有部分接口兼容未实现,所以pjdfstest用例进行部分裁剪,请使用下述源码进行验证
|
||||
$ git clone https://gitee.com/anar/pjdfstest.git
|
||||
|
||||
$ cd pjdfstest
|
||||
|
||||
# pjdfstest源码编译,生成prove二进制
|
||||
$ autoreconf-ifs
|
||||
|
||||
$ ./configure
|
||||
|
||||
$ make pjdfstest
|
||||
|
||||
```
|
||||
|
||||
### 测试
|
||||
|
||||
* 进入挂载qtfs文件系统的目录
|
||||
|
||||
* 执行prove -rv ${pjdfstest文件所在目录}
|
||||
|
||||
根据执行结果确定测试是否通过。
|
32
qtfs/test/qtfs_test/test_fifo_block.sh
Normal file
32
qtfs/test/qtfs_test/test_fifo_block.sh
Normal file
@ -0,0 +1,32 @@
|
||||
#!/bin/bash
|
||||
|
||||
# 检查参数
|
||||
if [ $# -ne 2 ]; then
|
||||
echo "Usage: $0 <thread_count> <path>"
|
||||
echo " thread_count: how many test thread to create."
|
||||
echo " path: qtfs path, script will create fifo in this path and test."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# 保存参数
|
||||
thread_count=$1
|
||||
path=$2
|
||||
|
||||
# 创建fifo文件
|
||||
for i in $(seq 1 $thread_count); do
|
||||
mkfifo "$path/test_fifo_block_$i"
|
||||
done
|
||||
|
||||
# 启动线程
|
||||
for i in $(seq 1 $thread_count); do
|
||||
(
|
||||
# 读取fifo文件
|
||||
read line < "$path/test_fifo_block_$i"
|
||||
echo "Thread $i read fifo: $line"
|
||||
# 删除fifo文件
|
||||
rm "$path/test_fifo_block_$i"
|
||||
) &
|
||||
done
|
||||
|
||||
# 等待所有线程结束
|
||||
wait
|
53
qtfs/test/qtfs_test/test_handle_open.rs
Normal file
53
qtfs/test/qtfs_test/test_handle_open.rs
Normal file
@ -0,0 +1,53 @@
|
||||
use std::io::{Write, Read};
|
||||
use std::net::TcpStream;
|
||||
use std::mem;
|
||||
use std::env;
|
||||
|
||||
#[repr(C, packed)]
|
||||
struct Qtreqopen {
|
||||
_type: u32,
|
||||
err: u32,
|
||||
seq_num: u64,
|
||||
len: usize,
|
||||
flags: u64,
|
||||
mode: u32,
|
||||
path: [u8; 4096],
|
||||
}
|
||||
|
||||
fn main() -> std::io::Result<()> {
|
||||
let argv: Vec<String> = env::args().collect();
|
||||
let addr: String = argv[1].parse().expect("please input addr(ip:port)");
|
||||
let mut stream = TcpStream::connect(addr).unwrap();
|
||||
let mut req = Qtreqopen {
|
||||
_type: 2,
|
||||
err: 0,
|
||||
seq_num: 0,
|
||||
len: 0,
|
||||
flags: 0,
|
||||
mode: 0,
|
||||
path: [0; 4096],
|
||||
};
|
||||
if argv.len() == 3 {
|
||||
let s: String = argv[2].parse().expect("Please input path to open by arg 2");
|
||||
req.path[..s.len()].copy_from_slice(s.as_bytes());
|
||||
req.len = s.len() + 12;
|
||||
} else {
|
||||
// send an err packet
|
||||
req.path = [1; 4096];
|
||||
req.len = 4108;
|
||||
}
|
||||
let bytes = unsafe {
|
||||
let ptr = &req as *const Qtreqopen as *const u8;
|
||||
std::slice::from_raw_parts(ptr, mem::size_of::<Qtreqopen>())
|
||||
};
|
||||
stream.write_all(&bytes[..24+req.len]).expect("failed to write to socket stream");
|
||||
// recv rsp
|
||||
let mut datain: [u8; 32] = [0; 32];
|
||||
stream.read(&mut datain).unwrap();
|
||||
if datain[28] == 1 || datain[4] == 1 {
|
||||
println!("Open failed!");
|
||||
} else {
|
||||
println!("Open successed.");
|
||||
}
|
||||
Ok(())
|
||||
}
|
23
qtfs/test/rexec_test/test_rexec_pipe_inherit.c
Normal file
23
qtfs/test/rexec_test/test_rexec_pipe_inherit.c
Normal file
@ -0,0 +1,23 @@
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
int fd[2];
|
||||
pipe(fd);
|
||||
int pid = fork();
|
||||
if (pid == 0) {
|
||||
// child
|
||||
char fdstr[16] = {0};
|
||||
sprintf(fdstr, "%d", fd[0]);
|
||||
char *argvchild[] = {"rexec", argv[1], fdstr, NULL};
|
||||
close(fd[1]);
|
||||
execv("/usr/bin/rexec", argvchild);
|
||||
perror("execv");
|
||||
}
|
||||
close(fd[0]);
|
||||
sleep(1);
|
||||
write(fd[1], "hello", 5);
|
||||
return 0;
|
||||
}
|
||||
|
13
qtfs/test/rexec_test/test_rexec_pipe_inherit_remote.c
Normal file
13
qtfs/test/rexec_test/test_rexec_pipe_inherit_remote.c
Normal file
@ -0,0 +1,13 @@
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <errno.h>
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
char buf[16] = {0};
|
||||
int fd = atoi(argv[1]);
|
||||
int ret = read(fd, buf, 16);
|
||||
printf("read from pipe fd:%d string:%s ret:%d errno:%d\n", fd, buf, ret, errno);
|
||||
return 0;
|
||||
}
|
||||
|
39
qtfs/test/scmright_test/README.md
Normal file
39
qtfs/test/scmright_test/README.md
Normal file
@ -0,0 +1,39 @@
|
||||
# 编译:
|
||||
```bash
|
||||
make clean
|
||||
make
|
||||
```
|
||||
|
||||
# 测试步骤:
|
||||
1. 在server上运行
|
||||
```bash
|
||||
cd /home
|
||||
tar xzvf scmright_test.tar.gz
|
||||
cd scmright_test
|
||||
make clean;make
|
||||
LD_LIBRARY_PATH=. ./server scm.sock
|
||||
```
|
||||
|
||||
2. 在client上运行
|
||||
```bash
|
||||
cd /home
|
||||
tar xzvf scmright_test.tar.gz
|
||||
cd scmright_test
|
||||
make clean;make
|
||||
LD_PRELOAD=/usr/lib64/libudsproxy.so LD_LIBRARY_PATH=. ./client scm.sock test.log
|
||||
or
|
||||
LD_PRELOAD=/usr/lib64/libudsproxy.so LD_LIBRARY_PATH=. ./client scm.sock
|
||||
```
|
||||
|
||||
3. 往client端发送消息进行测试
|
||||
```bash
|
||||
Input message to send:1234
|
||||
Input message to send:abcd
|
||||
Input message to send:quit
|
||||
input quit will exit client and server
|
||||
```
|
||||
|
||||
4. 确认server是否接收到消息或确认test.log是否写入
|
||||
```bash
|
||||
cat test.log | grep 1234
|
||||
```
|
81
qtfs/test/scmright_test/client.c
Normal file
81
qtfs/test/scmright_test/client.c
Normal file
@ -0,0 +1,81 @@
|
||||
#include <sys/socket.h>
|
||||
#include <sys/un.h>
|
||||
#include <unistd.h>
|
||||
#include <errno.h>
|
||||
#include <stddef.h>
|
||||
#include <stdio.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <fcntl.h>
|
||||
#include <stdbool.h>
|
||||
#include <linux/limits.h>
|
||||
#include <string.h>
|
||||
#include "uds.h"
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
if (argc < 2) {
|
||||
printf("Usage: %s pathtosock [pathtofile]\n", argv[0]);
|
||||
return -EINVAL;
|
||||
}
|
||||
char *path = argv[1];
|
||||
char *testfile = NULL;
|
||||
if (argc == 3) {
|
||||
testfile = argv[2];
|
||||
}
|
||||
int fd = socket(AF_UNIX, SOCK_STREAM, 0);
|
||||
if (fd < 0) {
|
||||
printf("Fail to create socket: %s\n", strerror(errno));
|
||||
return -1;
|
||||
}
|
||||
struct sockaddr_un addr;
|
||||
addr.sun_family = AF_UNIX;
|
||||
memcpy(addr.sun_path, path, strlen(path) + 1);
|
||||
printf("connect to %s\n", addr.sun_path);
|
||||
int ret = connect(fd, (struct sockaddr*)(&addr), sizeof(addr));
|
||||
if (ret < 0) {
|
||||
printf("Fail to connect: %s\n", strerror(errno));
|
||||
close(fd);
|
||||
return -1;
|
||||
}
|
||||
unlink(testfile);
|
||||
int share_fd;
|
||||
int pipefd[2];
|
||||
char buffer[1024];
|
||||
if (testfile == NULL) {
|
||||
printf("Into PIPE MODE: open pipe\n");
|
||||
if (pipe(pipefd) == -1) {
|
||||
fprintf(stderr, "Pipe failed\n");
|
||||
return -1;
|
||||
}
|
||||
share_fd = pipefd[1];
|
||||
} else {
|
||||
printf("Into REGULAR FILE MODE: open file %s\n", testfile);
|
||||
share_fd = open(testfile, O_RDWR | O_CREAT);
|
||||
}
|
||||
|
||||
if (share_fd < 0) {
|
||||
printf("open failed, %s\n", strerror(errno));
|
||||
return -1;
|
||||
}
|
||||
printf("send share_fd[%d] to peer\n", share_fd);
|
||||
udsSendFd(fd, share_fd);
|
||||
char data[PATH_MAX];
|
||||
while (true) {
|
||||
printf("Input message to send: ");
|
||||
gets(data);
|
||||
if (strncmp(data,"quit",4) == 0) {
|
||||
break;
|
||||
}
|
||||
ret = write(fd, data, strlen(data));
|
||||
if (ret < 0) {
|
||||
printf("Fail to write,%s\n", strerror(errno));
|
||||
}
|
||||
if(testfile == NULL && read(pipefd[0], buffer, sizeof(buffer))) {
|
||||
printf("read from pipe: %s\n", buffer);
|
||||
}
|
||||
|
||||
}
|
||||
close(fd);
|
||||
return 0;
|
||||
}
|
15
qtfs/test/scmright_test/makefile
Normal file
15
qtfs/test/scmright_test/makefile
Normal file
@ -0,0 +1,15 @@
|
||||
.PHONY: all
|
||||
|
||||
all: libscm.so client server
|
||||
|
||||
client: libscm.so client.c
|
||||
gcc client.c -L. -lscm -o client
|
||||
|
||||
server: libscm.so server.c
|
||||
gcc server.c -L. -lscm -o server
|
||||
|
||||
libscm.so: scm_rights.c
|
||||
gcc -fPIC --share scm_rights.c -o libscm.so
|
||||
|
||||
clean:
|
||||
rm -f client server libscm.so *.o
|
96
qtfs/test/scmright_test/scm_rights.c
Normal file
96
qtfs/test/scmright_test/scm_rights.c
Normal file
@ -0,0 +1,96 @@
|
||||
#include <stdio.h>
|
||||
# include <sys/socket.h>
|
||||
# include <sys/ioctl.h>
|
||||
# include <arpa/inet.h>
|
||||
# include <netinet/ip.h>
|
||||
# include <netinet/in.h>
|
||||
# include <netinet/udp.h>
|
||||
# include <netinet/tcp.h>
|
||||
# include <sys/un.h>
|
||||
# include <netdb.h>
|
||||
#include <fcntl.h>
|
||||
#include <errno.h>
|
||||
int
|
||||
udsRecvFd(int sock, int fdflags)
|
||||
{
|
||||
int byte = 0;
|
||||
struct iovec iov;
|
||||
struct msghdr msg;
|
||||
int fd = -1;
|
||||
ssize_t len;
|
||||
struct cmsghdr *cmsg;
|
||||
char buf[CMSG_SPACE(sizeof(fd))];
|
||||
int fdflags_recvmsg = fdflags & O_CLOEXEC ? MSG_CMSG_CLOEXEC : 0;
|
||||
|
||||
if ((fdflags & ~O_CLOEXEC) != 0) {
|
||||
errno = EINVAL;
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* send at least one char */
|
||||
memset(&msg, 0, sizeof(msg));
|
||||
iov.iov_base = &byte;
|
||||
iov.iov_len = 4;
|
||||
msg.msg_iov = &iov;
|
||||
msg.msg_iovlen = 1;
|
||||
msg.msg_name = NULL;
|
||||
msg.msg_namelen = 0;
|
||||
|
||||
msg.msg_control = buf;
|
||||
msg.msg_controllen = sizeof(buf);
|
||||
cmsg = CMSG_FIRSTHDR(&msg);
|
||||
cmsg->cmsg_level = SOL_SOCKET;
|
||||
cmsg->cmsg_len = CMSG_LEN(sizeof(fd));
|
||||
/* Initialize the payload: */
|
||||
memcpy(CMSG_DATA(cmsg), &fd, sizeof(fd));
|
||||
msg.msg_controllen = cmsg->cmsg_len;
|
||||
|
||||
len = recvmsg(sock, &msg, fdflags_recvmsg);
|
||||
if (len < 0)
|
||||
return -1;
|
||||
|
||||
cmsg = CMSG_FIRSTHDR(&msg);
|
||||
printf("recv msg type:%d SCM_RIGHTS:%d, byte:%d\n", cmsg->cmsg_type, SCM_RIGHTS, byte);
|
||||
/* be paranoiac */
|
||||
if (len == 0 || cmsg == NULL || cmsg->cmsg_len != CMSG_LEN(sizeof(fd))
|
||||
|| cmsg->cmsg_level != SOL_SOCKET || cmsg->cmsg_type != SCM_RIGHTS) {
|
||||
/* fake errno: at end the file is not available */
|
||||
errno = len ? EACCES : ENOTCONN;
|
||||
return -1;
|
||||
}
|
||||
|
||||
memcpy(&fd, CMSG_DATA(cmsg), sizeof(fd));
|
||||
|
||||
return fd;
|
||||
}
|
||||
int udsSendFd(int sock, int fd)
|
||||
{
|
||||
char byte = 0;
|
||||
struct iovec iov;
|
||||
struct msghdr msg;
|
||||
struct cmsghdr *cmsg;
|
||||
char buf[CMSG_SPACE(sizeof(fd))];
|
||||
|
||||
/* send at least one char */
|
||||
memset(&msg, 0, sizeof(msg));
|
||||
iov.iov_base = &byte;
|
||||
iov.iov_len = 1;
|
||||
msg.msg_iov = &iov;
|
||||
msg.msg_iovlen = 1;
|
||||
msg.msg_name = NULL;
|
||||
msg.msg_namelen = 0;
|
||||
|
||||
msg.msg_control = buf;
|
||||
msg.msg_controllen = sizeof(buf);
|
||||
cmsg = CMSG_FIRSTHDR(&msg);
|
||||
cmsg->cmsg_level = SOL_SOCKET;
|
||||
cmsg->cmsg_type = SCM_RIGHTS;
|
||||
cmsg->cmsg_len = CMSG_LEN(sizeof(fd));
|
||||
/* Initialize the payload: */
|
||||
memcpy(CMSG_DATA(cmsg), &fd, sizeof(fd));
|
||||
msg.msg_controllen = cmsg->cmsg_len;
|
||||
|
||||
if (sendmsg(sock, &msg, 0) != iov.iov_len)
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
88
qtfs/test/scmright_test/server.c
Normal file
88
qtfs/test/scmright_test/server.c
Normal file
@ -0,0 +1,88 @@
|
||||
#include <errno.h>
|
||||
#include <netinet/in.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/un.h>
|
||||
#include <unistd.h>
|
||||
#include <stddef.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <fcntl.h>
|
||||
#include <linux/limits.h>
|
||||
#include "uds.h"
|
||||
|
||||
static const char *HEAD = "write from server:\n";
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
char *path = NULL;
|
||||
if (argc != 2) {
|
||||
printf("Usage: %s pathtosock\n", argv[0]);
|
||||
return -EINVAL;
|
||||
}
|
||||
path=argv[1];
|
||||
int fd = socket(AF_UNIX, SOCK_STREAM, 0);
|
||||
if (fd < 0) {
|
||||
printf("Fail to create socket: %s\n", strerror(errno));
|
||||
return -1;
|
||||
}
|
||||
struct sockaddr_un addr;
|
||||
addr.sun_family = AF_UNIX;
|
||||
memcpy(addr.sun_path, path, strlen(path) + 1);
|
||||
socklen_t size = offsetof(struct sockaddr_un, sun_path) + strlen(path) + 1;;
|
||||
unlink(path);
|
||||
printf("begin bind %s\n", addr.sun_path);
|
||||
int ret = bind(fd, (struct sockaddr*)&addr, sizeof(addr));
|
||||
if (ret < 0) {
|
||||
printf("Fail to bind socket: %s\n", strerror(errno));
|
||||
return ret;
|
||||
}
|
||||
printf("begin listen...\n");
|
||||
ret = listen(fd, 10);
|
||||
if (ret < 0) {
|
||||
printf("Fail to listen, %s\n", strerror(errno));
|
||||
close(fd);
|
||||
return ret;
|
||||
}
|
||||
size = sizeof(addr);
|
||||
while (1) {
|
||||
printf("try accept %s...\n", addr.sun_path);
|
||||
int new_fd = accept(fd, (struct sockaddr*)&addr, &size);
|
||||
if (new_fd < 0) {
|
||||
printf("Fail to accpet, %s\n", strerror(errno));
|
||||
close(fd);
|
||||
return -1;
|
||||
}
|
||||
printf("try udsRecvFd...\n");
|
||||
int share_fd = udsRecvFd(new_fd, 0);
|
||||
if (share_fd) {
|
||||
printf("server write to share_fd[%d]\n", share_fd);
|
||||
write(share_fd, HEAD, strlen(HEAD));
|
||||
}
|
||||
printf("receive share_fd=%d\n", share_fd);
|
||||
char buf[PATH_MAX];
|
||||
while (1) {
|
||||
printf("try read from uds client...\n");
|
||||
ret = read(new_fd, buf, PATH_MAX);
|
||||
if (ret <= 0) {
|
||||
printf("closed: %s\n", strerror(errno));
|
||||
close(fd);
|
||||
close(new_fd);
|
||||
unlink(path);
|
||||
return -1;
|
||||
}
|
||||
buf[ret] = '\0';
|
||||
if (strncmp(buf, "quit", 4) == 0) {
|
||||
printf("receive quit\n");
|
||||
break;
|
||||
}
|
||||
printf("readmsg from client: %s\n", buf);
|
||||
write(share_fd, buf, ret);
|
||||
}
|
||||
close(share_fd);
|
||||
close(new_fd);
|
||||
}
|
||||
close(fd);
|
||||
}
|
9
qtfs/test/scmright_test/uds.h
Normal file
9
qtfs/test/scmright_test/uds.h
Normal file
@ -0,0 +1,9 @@
|
||||
#ifndef __UDS_H__
|
||||
#define __UDS_H__
|
||||
|
||||
|
||||
int udsRecvFd(int sock, int fdflags);
|
||||
|
||||
int udsSendFd(int sock, int fd);
|
||||
|
||||
#endif
|
340
qtfs/utils/License
Normal file
340
qtfs/utils/License
Normal file
@ -0,0 +1,340 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 2, June 1991
|
||||
|
||||
Copyright (C) 1989, 1991 Free Software Foundation, Inc.
|
||||
51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The licenses for most software are designed to take away your
|
||||
freedom to share and change it. By contrast, the GNU General Public
|
||||
License is intended to guarantee your freedom to share and change free
|
||||
software--to make sure the software is free for all its users. This
|
||||
General Public License applies to most of the Free Software
|
||||
Foundation's software and to any other program whose authors commit to
|
||||
using it. (Some other Free Software Foundation software is covered by
|
||||
the GNU Library General Public License instead.) You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
this service if you wish), that you receive source code or can get it
|
||||
if you want it, that you can change the software or use pieces of it
|
||||
in new free programs; and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to make restrictions that forbid
|
||||
anyone to deny you these rights or to ask you to surrender the rights.
|
||||
These restrictions translate to certain responsibilities for you if you
|
||||
distribute copies of the software, or if you modify it.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must give the recipients all the rights that
|
||||
you have. You must make sure that they, too, receive or can get the
|
||||
source code. And you must show them these terms so they know their
|
||||
rights.
|
||||
|
||||
We protect your rights with two steps: (1) copyright the software, and
|
||||
(2) offer you this license which gives you legal permission to copy,
|
||||
distribute and/or modify the software.
|
||||
|
||||
Also, for each author's protection and ours, we want to make certain
|
||||
that everyone understands that there is no warranty for this free
|
||||
software. If the software is modified by someone else and passed on, we
|
||||
want its recipients to know that what they have is not the original, so
|
||||
that any problems introduced by others will not reflect on the original
|
||||
authors' reputations.
|
||||
|
||||
Finally, any free program is threatened constantly by software
|
||||
patents. We wish to avoid the danger that redistributors of a free
|
||||
program will individually obtain patent licenses, in effect making the
|
||||
program proprietary. To prevent this, we have made it clear that any
|
||||
patent must be licensed for everyone's free use or not licensed at all.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||
|
||||
0. This License applies to any program or other work which contains
|
||||
a notice placed by the copyright holder saying it may be distributed
|
||||
under the terms of this General Public License. The "Program", below,
|
||||
refers to any such program or work, and a "work based on the Program"
|
||||
means either the Program or any derivative work under copyright law:
|
||||
that is to say, a work containing the Program or a portion of it,
|
||||
either verbatim or with modifications and/or translated into another
|
||||
language. (Hereinafter, translation is included without limitation in
|
||||
the term "modification".) Each licensee is addressed as "you".
|
||||
|
||||
Activities other than copying, distribution and modification are not
|
||||
covered by this License; they are outside its scope. The act of
|
||||
running the Program is not restricted, and the output from the Program
|
||||
is covered only if its contents constitute a work based on the
|
||||
Program (independent of having been made by running the Program).
|
||||
Whether that is true depends on what the Program does.
|
||||
|
||||
1. You may copy and distribute verbatim copies of the Program's
|
||||
source code as you receive it, in any medium, provided that you
|
||||
conspicuously and appropriately publish on each copy an appropriate
|
||||
copyright notice and disclaimer of warranty; keep intact all the
|
||||
notices that refer to this License and to the absence of any warranty;
|
||||
and give any other recipients of the Program a copy of this License
|
||||
along with the Program.
|
||||
|
||||
You may charge a fee for the physical act of transferring a copy, and
|
||||
you may at your option offer warranty protection in exchange for a fee.
|
||||
|
||||
2. You may modify your copy or copies of the Program or any portion
|
||||
of it, thus forming a work based on the Program, and copy and
|
||||
distribute such modifications or work under the terms of Section 1
|
||||
above, provided that you also meet all of these conditions:
|
||||
|
||||
a) You must cause the modified files to carry prominent notices
|
||||
stating that you changed the files and the date of any change.
|
||||
|
||||
b) You must cause any work that you distribute or publish, that in
|
||||
whole or in part contains or is derived from the Program or any
|
||||
part thereof, to be licensed as a whole at no charge to all third
|
||||
parties under the terms of this License.
|
||||
|
||||
c) If the modified program normally reads commands interactively
|
||||
when run, you must cause it, when started running for such
|
||||
interactive use in the most ordinary way, to print or display an
|
||||
announcement including an appropriate copyright notice and a
|
||||
notice that there is no warranty (or else, saying that you provide
|
||||
a warranty) and that users may redistribute the program under
|
||||
these conditions, and telling the user how to view a copy of this
|
||||
License. (Exception: if the Program itself is interactive but
|
||||
does not normally print such an announcement, your work based on
|
||||
the Program is not required to print an announcement.)
|
||||
|
||||
These requirements apply to the modified work as a whole. If
|
||||
identifiable sections of that work are not derived from the Program,
|
||||
and can be reasonably considered independent and separate works in
|
||||
themselves, then this License, and its terms, do not apply to those
|
||||
sections when you distribute them as separate works. But when you
|
||||
distribute the same sections as part of a whole which is a work based
|
||||
on the Program, the distribution of the whole must be on the terms of
|
||||
this License, whose permissions for other licensees extend to the
|
||||
entire whole, and thus to each and every part regardless of who wrote it.
|
||||
|
||||
Thus, it is not the intent of this section to claim rights or contest
|
||||
your rights to work written entirely by you; rather, the intent is to
|
||||
exercise the right to control the distribution of derivative or
|
||||
collective works based on the Program.
|
||||
|
||||
In addition, mere aggregation of another work not based on the Program
|
||||
with the Program (or with a work based on the Program) on a volume of
|
||||
a storage or distribution medium does not bring the other work under
|
||||
the scope of this License.
|
||||
|
||||
3. You may copy and distribute the Program (or a work based on it,
|
||||
under Section 2) in object code or executable form under the terms of
|
||||
Sections 1 and 2 above provided that you also do one of the following:
|
||||
|
||||
a) Accompany it with the complete corresponding machine-readable
|
||||
source code, which must be distributed under the terms of Sections
|
||||
1 and 2 above on a medium customarily used for software interchange; or,
|
||||
|
||||
b) Accompany it with a written offer, valid for at least three
|
||||
years, to give any third party, for a charge no more than your
|
||||
cost of physically performing source distribution, a complete
|
||||
machine-readable copy of the corresponding source code, to be
|
||||
distributed under the terms of Sections 1 and 2 above on a medium
|
||||
customarily used for software interchange; or,
|
||||
|
||||
c) Accompany it with the information you received as to the offer
|
||||
to distribute corresponding source code. (This alternative is
|
||||
allowed only for noncommercial distribution and only if you
|
||||
received the program in object code or executable form with such
|
||||
an offer, in accord with Subsection b above.)
|
||||
|
||||
The source code for a work means the preferred form of the work for
|
||||
making modifications to it. For an executable work, complete source
|
||||
code means all the source code for all modules it contains, plus any
|
||||
associated interface definition files, plus the scripts used to
|
||||
control compilation and installation of the executable. However, as a
|
||||
special exception, the source code distributed need not include
|
||||
anything that is normally distributed (in either source or binary
|
||||
form) with the major components (compiler, kernel, and so on) of the
|
||||
operating system on which the executable runs, unless that component
|
||||
itself accompanies the executable.
|
||||
|
||||
If distribution of executable or object code is made by offering
|
||||
access to copy from a designated place, then offering equivalent
|
||||
access to copy the source code from the same place counts as
|
||||
distribution of the source code, even though third parties are not
|
||||
compelled to copy the source along with the object code.
|
||||
|
||||
4. You may not copy, modify, sublicense, or distribute the Program
|
||||
except as expressly provided under this License. Any attempt
|
||||
otherwise to copy, modify, sublicense or distribute the Program is
|
||||
void, and will automatically terminate your rights under this License.
|
||||
However, parties who have received copies, or rights, from you under
|
||||
this License will not have their licenses terminated so long as such
|
||||
parties remain in full compliance.
|
||||
|
||||
5. You are not required to accept this License, since you have not
|
||||
signed it. However, nothing else grants you permission to modify or
|
||||
distribute the Program or its derivative works. These actions are
|
||||
prohibited by law if you do not accept this License. Therefore, by
|
||||
modifying or distributing the Program (or any work based on the
|
||||
Program), you indicate your acceptance of this License to do so, and
|
||||
all its terms and conditions for copying, distributing or modifying
|
||||
the Program or works based on it.
|
||||
|
||||
6. Each time you redistribute the Program (or any work based on the
|
||||
Program), the recipient automatically receives a license from the
|
||||
original licensor to copy, distribute or modify the Program subject to
|
||||
these terms and conditions. You may not impose any further
|
||||
restrictions on the recipients' exercise of the rights granted herein.
|
||||
You are not responsible for enforcing compliance by third parties to
|
||||
this License.
|
||||
|
||||
7. If, as a consequence of a court judgment or allegation of patent
|
||||
infringement or for any other reason (not limited to patent issues),
|
||||
conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot
|
||||
distribute so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you
|
||||
may not distribute the Program at all. For example, if a patent
|
||||
license would not permit royalty-free redistribution of the Program by
|
||||
all those who receive copies directly or indirectly through you, then
|
||||
the only way you could satisfy both it and this License would be to
|
||||
refrain entirely from distribution of the Program.
|
||||
|
||||
If any portion of this section is held invalid or unenforceable under
|
||||
any particular circumstance, the balance of the section is intended to
|
||||
apply and the section as a whole is intended to apply in other
|
||||
circumstances.
|
||||
|
||||
It is not the purpose of this section to induce you to infringe any
|
||||
patents or other property right claims or to contest validity of any
|
||||
such claims; this section has the sole purpose of protecting the
|
||||
integrity of the free software distribution system, which is
|
||||
implemented by public license practices. Many people have made
|
||||
generous contributions to the wide range of software distributed
|
||||
through that system in reliance on consistent application of that
|
||||
system; it is up to the author/donor to decide if he or she is willing
|
||||
to distribute software through any other system and a licensee cannot
|
||||
impose that choice.
|
||||
|
||||
This section is intended to make thoroughly clear what is believed to
|
||||
be a consequence of the rest of this License.
|
||||
|
||||
8. If the distribution and/or use of the Program is restricted in
|
||||
certain countries either by patents or by copyrighted interfaces, the
|
||||
original copyright holder who places the Program under this License
|
||||
may add an explicit geographical distribution limitation excluding
|
||||
those countries, so that distribution is permitted only in or among
|
||||
countries not thus excluded. In such case, this License incorporates
|
||||
the limitation as if written in the body of this License.
|
||||
|
||||
9. The Free Software Foundation may publish revised and/or new versions
|
||||
of the General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the Program
|
||||
specifies a version number of this License which applies to it and "any
|
||||
later version", you have the option of following the terms and conditions
|
||||
either of that version or of any later version published by the Free
|
||||
Software Foundation. If the Program does not specify a version number of
|
||||
this License, you may choose any version ever published by the Free Software
|
||||
Foundation.
|
||||
|
||||
10. If you wish to incorporate parts of the Program into other free
|
||||
programs whose distribution conditions are different, write to the author
|
||||
to ask for permission. For software which is copyrighted by the Free
|
||||
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||
make exceptions for this. Our decision will be guided by the two goals
|
||||
of preserving the free status of all derivatives of our free software and
|
||||
of promoting the sharing and reuse of software generally.
|
||||
|
||||
NO WARRANTY
|
||||
|
||||
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||
REPAIR OR CORRECTION.
|
||||
|
||||
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||
POSSIBILITY OF SUCH DAMAGES.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
convey the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation; either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program; if not, write to the Free Software
|
||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program is interactive, make it output a short notice like this
|
||||
when it starts in an interactive mode:
|
||||
|
||||
Gnomovision version 69, Copyright (C) year name of author
|
||||
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, the commands you use may
|
||||
be called something other than `show w' and `show c'; they could even be
|
||||
mouse-clicks or menu items--whatever suits your program.
|
||||
|
||||
You should also get your employer (if you work as a programmer) or your
|
||||
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||
necessary. Here is a sample; alter the names:
|
||||
|
||||
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||
|
||||
<signature of Ty Coon>, 1 April 1989
|
||||
Ty Coon, President of Vice
|
||||
|
||||
This General Public License does not permit incorporating your program into
|
||||
proprietary programs. If your program is a subroutine library, you may
|
||||
consider it more useful to permit linking proprietary applications with the
|
||||
library. If this is what you want to do, use the GNU Library General
|
||||
Public License instead of this License.
|
55
qtfs/utils/qtfs_utils.h
Normal file
55
qtfs/utils/qtfs_utils.h
Normal file
@ -0,0 +1,55 @@
|
||||
/******************************************************************************
|
||||
* Copyright (c) Huawei Technologies Co., Ltd. 2023. All rights reserved.
|
||||
* qtfs licensed under the Mulan PSL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
||||
* You may obtain a copy of Mulan PSL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPSL2
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||||
* PURPOSE.
|
||||
* See the Mulan PSL v2 for more details.
|
||||
* Author: Liqiang
|
||||
* Create: 2023-03-20
|
||||
* Description:
|
||||
*******************************************************************************/
|
||||
|
||||
#ifndef __QTFS_UTILS_H__
|
||||
#define __QTFS_UTILS_H__
|
||||
|
||||
#define QTFS_UTILS_DEV "/dev/qtfs_utils"
|
||||
|
||||
// QTFS provide some remote capability
|
||||
#define QTUTIL_IOCTL_CAPA_MAGIC 'U'
|
||||
enum {
|
||||
_QTUTIL_IOCTL_CAPA_PORT_INUSE,
|
||||
};
|
||||
#define QTUTIL_CAPA(CAP) _IO(QTUTIL_IOCTL_CAPA_MAGIC, CAP)
|
||||
#define QTUTIL_CAPA_PORT_INUSE QTUTIL_CAPA(_QTUTIL_IOCTL_CAPA_PORT_INUSE)
|
||||
|
||||
// QTFS provide some remote syscalls
|
||||
#define QTUTIL_IOCTL_SC_MAGIC 'S'
|
||||
enum {
|
||||
_QTUTIL_IOCTL_SYSCALL_KILL,
|
||||
_QTUTIL_IOCTL_SYSCALL_SCHED_SETAFFINITY,
|
||||
_QTUTIL_IOCTL_SYSCALL_SCHED_GETAFFINITY,
|
||||
};
|
||||
#define QTUTIL_SYSCALL(SC) _IO(QTUTIL_IOCTL_SC_MAGIC, SC)
|
||||
#define QTUTIL_SC_KILL QTUTIL_SYSCALL(_QTUTIL_IOCTL_SYSCALL_KILL)
|
||||
#define QTUTIL_SC_SCHED_SETAFFINITY QTUTIL_SYSCALL(_QTUTIL_IOCTL_SYSCALL_SCHED_SETAFFINITY)
|
||||
#define QTUTIL_SC_SCHED_GETAFFINITY QTUTIL_SYSCALL(_QTUTIL_IOCTL_SYSCALL_SCHED_GETAFFINITY)
|
||||
|
||||
struct qtsc_kill {
|
||||
int pid;
|
||||
int signum;
|
||||
};
|
||||
|
||||
// sched getaffinity and set affinity
|
||||
struct qtsc_sched_affinity {
|
||||
int pid;
|
||||
unsigned int len;
|
||||
unsigned long *user_mask_ptr;
|
||||
};
|
||||
|
||||
|
||||
#endif
|
||||
|
202
qtfs/utils/utils.c
Normal file
202
qtfs/utils/utils.c
Normal file
@ -0,0 +1,202 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Copyright (C) 2023. Huawei Technologies Co., Ltd. All rights reserved.
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 and
|
||||
* only version 2 as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*/
|
||||
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/fs.h>
|
||||
#include <linux/miscdevice.h>
|
||||
#include <linux/mm.h>
|
||||
#include <linux/time.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/uaccess.h>
|
||||
|
||||
#include "conn.h"
|
||||
#include "log.h"
|
||||
#include "qtfs_utils.h"
|
||||
#include "req.h"
|
||||
#include "qtfs-mod.h"
|
||||
|
||||
#ifndef QTFS_CLIENT
|
||||
#error "QTFS utils only use in qtfs client."
|
||||
#endif
|
||||
|
||||
long qtfs_utils_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
long qtfs_capability_ioctl(struct qtfs_conn_var_s *pvar, struct file *file, unsigned int cmd, unsigned long arg);
|
||||
long qtfs_syscall_ioctl(struct qtfs_conn_var_s *pvar, struct file *file, unsigned int cmd, unsigned long arg);
|
||||
|
||||
struct file_operations qtfs_utils_fops = {
|
||||
.owner = THIS_MODULE,
|
||||
.unlocked_ioctl = qtfs_utils_ioctl,
|
||||
};
|
||||
|
||||
static struct miscdevice qtfs_utils_dev = {
|
||||
.minor = MISC_DYNAMIC_MINOR,
|
||||
.name = "qtfs_utils",
|
||||
.fops = &qtfs_utils_fops,
|
||||
};
|
||||
|
||||
int qtfs_utils_register(void)
|
||||
{
|
||||
int ret = misc_register(&qtfs_utils_dev);
|
||||
|
||||
if (ret) {
|
||||
qtfs_err("qtfs utils dev register failed, ret:%d", ret);
|
||||
return -EFAULT;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
void qtfs_utils_destroy(void)
|
||||
{
|
||||
misc_deregister(&qtfs_utils_dev);
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
long qtfs_utils_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
|
||||
{
|
||||
long ret = -EINVAL;
|
||||
struct qtfs_conn_var_s *pvar = qtfs_conn_get_param();
|
||||
|
||||
if (pvar == NULL) {
|
||||
qtfs_err("utils ioctl get pvar failed, cmd type:%c nr:%d", _IOC_TYPE(cmd), _IOC_NR(cmd));
|
||||
return -EFAULT;
|
||||
}
|
||||
switch (_IOC_TYPE(cmd)) {
|
||||
case QTUTIL_IOCTL_CAPA_MAGIC:
|
||||
ret = qtfs_capability_ioctl(pvar, file, cmd, arg);
|
||||
qtfs_info("utils capability ioctl nr:%d", _IOC_NR(cmd));
|
||||
break;
|
||||
|
||||
case QTUTIL_IOCTL_SC_MAGIC:
|
||||
ret = qtfs_syscall_ioctl(pvar, file, cmd, arg);
|
||||
qtfs_info("utils syscall ioctl nr:%d", _IOC_NR(cmd));
|
||||
break;
|
||||
|
||||
default:
|
||||
qtfs_err("Unsupport magic type:%c", _IOC_TYPE(cmd));
|
||||
break;
|
||||
}
|
||||
qtfs_conn_put_param(pvar);
|
||||
return ret;
|
||||
}
|
||||
|
||||
long qtfs_capability_ioctl(struct qtfs_conn_var_s *pvar, struct file *file, unsigned int cmd, unsigned long arg)
|
||||
{
|
||||
long ret = -ENOTSUPP;
|
||||
return ret;
|
||||
}
|
||||
|
||||
static long qtfs_sc_kill(struct qtfs_conn_var_s *pvar, unsigned long arg)
|
||||
{
|
||||
struct qtreq_sc_kill *req;
|
||||
struct qtrsp_sc_kill *rsp;
|
||||
struct qtsc_kill karg;
|
||||
|
||||
if (copy_from_user(&karg, (void *)arg, sizeof(struct qtsc_kill))) {
|
||||
qtfs_err("copy args failed.");
|
||||
return -EINVAL;
|
||||
}
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
req->pid = karg.pid;
|
||||
req->signum = karg.signum;
|
||||
rsp = qtfs_remote_run(pvar, QTFS_SC_KILL, sizeof(struct qtreq_sc_kill));
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
qtfs_err("qtfs remote kill faile.");
|
||||
return -EFAULT;
|
||||
}
|
||||
qtfs_info("qtfs remote kill pid:%d sig:%d success:%ld", req->pid, req->signum, rsp->ret);
|
||||
return rsp->ret;
|
||||
}
|
||||
|
||||
static long qtfs_sc_setaffinity(struct qtfs_conn_var_s *pvar, unsigned long arg)
|
||||
{
|
||||
struct qtreq_sc_sched_affinity *req;
|
||||
struct qtrsp_sc_sched_affinity *rsp;
|
||||
struct qtsc_sched_affinity karg;
|
||||
|
||||
if (copy_from_user(&karg, (void *)arg, sizeof(struct qtsc_sched_affinity))) {
|
||||
qtfs_err("copy args failed.");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
req->type = SC_SET;
|
||||
req->pid = karg.pid;
|
||||
req->len = (karg.len > AFFINITY_MAX_LEN) ? AFFINITY_MAX_LEN : karg.len;
|
||||
if (copy_from_user(req->user_mask_ptr, karg.user_mask_ptr, req->len)) {
|
||||
qtfs_err("copy from user mask ptr failed, len:%u", karg.len);
|
||||
return -EFAULT;
|
||||
}
|
||||
rsp = qtfs_remote_run(pvar, QTFS_SC_SCHED_SETAFFINITY, sizeof(struct qtreq_sc_sched_affinity) + karg.len * sizeof(unsigned long));
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
qtfs_err("qtfs remote set affinit failed, pid:%d len:%u", karg.pid, karg.len);
|
||||
return -EFAULT;
|
||||
}
|
||||
qtfs_info("qtfs remote set affinity successed pid:%d len:%u return:%ld", karg.pid, karg.len, rsp->ret);
|
||||
return rsp->ret;
|
||||
}
|
||||
|
||||
static long qtfs_sc_getaffinity(struct qtfs_conn_var_s *pvar, unsigned long arg)
|
||||
{
|
||||
struct qtreq_sc_sched_affinity *req;
|
||||
struct qtrsp_sc_sched_affinity *rsp;
|
||||
struct qtsc_sched_affinity karg;
|
||||
|
||||
if (copy_from_user(&karg, (void *)arg, sizeof(struct qtsc_sched_affinity))) {
|
||||
qtfs_err("copy args failed.");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
req = pvar->conn_ops->get_conn_msg_buf(pvar, QTFS_SEND);
|
||||
req->type = SC_GET;
|
||||
req->pid = karg.pid;
|
||||
req->len = (karg.len > AFFINITY_MAX_LEN) ? AFFINITY_MAX_LEN : karg.len;
|
||||
rsp = qtfs_remote_run(pvar, QTFS_SC_SCHED_GETAFFINITY, sizeof(struct qtreq_sc_sched_affinity));
|
||||
if (IS_ERR_OR_NULL(rsp)) {
|
||||
qtfs_err("qtfs remote get affinit failed, pid:%d len:%u", karg.pid, karg.len);
|
||||
return -EFAULT;
|
||||
}
|
||||
// len == 0 means failed
|
||||
if (rsp->len <= 0 || rsp->len > req->len ||
|
||||
copy_to_user(karg.user_mask_ptr, rsp->user_mask_ptr, rsp->len)) {
|
||||
qtfs_err("copy user mask ptr failed rsp len:%u valid len:%u", rsp->len, karg.len);
|
||||
return -EINVAL;
|
||||
}
|
||||
return rsp->ret;
|
||||
}
|
||||
|
||||
long qtfs_syscall_ioctl(struct qtfs_conn_var_s *pvar, struct file *file, unsigned int cmd, unsigned long arg)
|
||||
{
|
||||
long ret = -EINVAL;
|
||||
|
||||
switch (cmd) {
|
||||
case QTUTIL_SC_KILL:
|
||||
ret = qtfs_sc_kill(pvar, arg);
|
||||
break;
|
||||
|
||||
case QTUTIL_SC_SCHED_SETAFFINITY:
|
||||
ret = qtfs_sc_setaffinity(pvar, arg);
|
||||
break;
|
||||
|
||||
case QTUTIL_SC_SCHED_GETAFFINITY:
|
||||
ret = qtfs_sc_getaffinity(pvar, arg);
|
||||
break;
|
||||
|
||||
default:
|
||||
qtfs_err("syscall ioctl not support magic:%c number:%d", _IOC_TYPE(cmd), _IOC_NR(cmd));
|
||||
break;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
66
script.md
Normal file
66
script.md
Normal file
@ -0,0 +1,66 @@
|
||||
### 第一幕 介绍环境信息
|
||||
1> 大禹/Host硬件信息。
|
||||
-- 略
|
||||
|
||||
2> 操作系统信息。
|
||||
+-- 命令1,HOST/DPU: cat /etc/os-release
|
||||
说明:查看OS发行版信息。
|
||||
|
||||
3> 仅dpu侧有libvirt,host上无关键进程。
|
||||
(逐步标出关键信息)
|
||||
第3步命令列表:
|
||||
命令1,DPU/HOST: ps -ef |grep libvirtd
|
||||
说明:在DPU上查询到libvirtd守护进程存在,而HOST上没有libvirt守护进程。
|
||||
|
||||
命令2,DPU/HOST: ps -ef |grep qemu
|
||||
说明:在DPU和HOST上初始状态均没有虚拟机相关进程存在。
|
||||
|
||||
### 第二幕 DPU侧
|
||||
DPU> 1> virsh define / start (字幕:创建虚拟机)操作
|
||||
第1步命令:
|
||||
命令1,DPU: virsh define /root/remote_vm.xml
|
||||
说明:从xml文件创建虚机配置。
|
||||
|
||||
命令2,DPU: virsh start remote_vm
|
||||
说明:启动刚刚创建的虚机remote_vm
|
||||
|
||||
DPU> 1.1 > virsh list(字幕:虚拟机已创建)结果
|
||||
第2步命令:
|
||||
命令1,DPU: virsh list
|
||||
说明:查询到libvirt管理的虚机,能查询到remote_vm的状态为running
|
||||
|
||||
|
||||
DPU> 2> ps查询进程状态,无qemu(但有proxy进程组件)结果
|
||||
第3步命令:
|
||||
命令1,DPU: ps -ef |grep qemu
|
||||
说明:能查询到一个/usr/bin/rexec进程带有qemu关键字的参数,这是一个虚机在DPU的代理进程,不是真正的虚机。
|
||||
|
||||
|
||||
|
||||
DPU> 4> virsh console查看虚拟机 (字幕:虚拟机可访问)结果
|
||||
第4步命令:
|
||||
命令1,DPU:virsh console remote_vm
|
||||
说明:账号输入“root”,密码输入“openEuler12#$”登入虚拟机。
|
||||
|
||||
命令2,在刚刚登入的虚机内进行一些简单的操作:
|
||||
cat /proc/cpuinfo |grep processor (查看虚机CPU核数)
|
||||
cat /proc/cpuinfo |grep QEMU (查看CPU MODEL信息)
|
||||
cat /proc/meminfo |grep MemTotal (查看虚机内存总量)
|
||||
|
||||
命令3,在登入的虚机中退出虚机控制台:ctrl + ]
|
||||
|
||||
|
||||
### 第三幕 HOST侧,
|
||||
HOST> 3> ps查询进程状态,有qemu 结果
|
||||
命令1,HOST: ps -ef |grep qemu
|
||||
说明:能查询到一个qemu-kvm真实虚机进程,查看其参数中有-name guest=remote_vm字样,说明真正的虚机运行在host。
|
||||
|
||||
第四幕 DPU侧删除虚拟机
|
||||
DPU> virsh destroy
|
||||
命令,DPU:virsh destroy remote_vm
|
||||
说明:关闭虚机。
|
||||
|
||||
DPU> virsh list
|
||||
命令,DPU:virsh list
|
||||
说明:没有running状态虚机显示,如果输入virsh list --all则显示remote_vm虚机为shut off状态
|
||||
|
110
virsh_demo.xml
Normal file
110
virsh_demo.xml
Normal file
@ -0,0 +1,110 @@
|
||||
<!--
|
||||
WARNING: THIS IS AN AUTO-GENERATED FILE. CHANGES TO IT ARE LIKELY TO BE
|
||||
OVERWRITTEN AND LOST. Changes to this xml configuration should be made using:
|
||||
virsh edit vm23
|
||||
or other application using the libvirt API.
|
||||
-->
|
||||
|
||||
<domain type='kvm'>
|
||||
<name>lq</name>
|
||||
<uuid>7c0189f5-457c-1834-9387-73de2860b088</uuid>
|
||||
<memory unit='KiB'>4194304</memory>
|
||||
<currentMemory unit='KiB'>4194304</currentMemory>
|
||||
<memtune>
|
||||
<hard_limit unit='KiB'>4194304</hard_limit>
|
||||
</memtune>
|
||||
<vcpu placement='static'>4</vcpu>
|
||||
<iothreads>3</iothreads>
|
||||
<iothreadids>
|
||||
<iothread id='1'/>
|
||||
<iothread id='2'/>
|
||||
<iothread id='3'/>
|
||||
</iothreadids>
|
||||
<os>
|
||||
<type arch='aarch64' machine='virt-5.0'>hvm</type>
|
||||
<loader readonly='yes' type='pflash'>/usr/share/edk2/aarch64/QEMU_EFI-pflash.raw</loader>
|
||||
<nvram>/var/lib/libvirt/qemu/nvram/common_VARS.fd</nvram>
|
||||
<boot dev='hd'/>
|
||||
</os>
|
||||
<features>
|
||||
<acpi/>
|
||||
<gic version='3'/>
|
||||
</features>
|
||||
<cpu mode='host-passthrough' check='none'/>
|
||||
<clock offset='utc'/>
|
||||
<on_poweroff>destroy</on_poweroff>
|
||||
<on_reboot>restart</on_reboot>
|
||||
<on_crash>destroy</on_crash>
|
||||
<devices>
|
||||
<emulator>/usr/bin/qemu-rexec</emulator>
|
||||
<disk type='file' device='disk'>
|
||||
<driver name='qemu' type='qcow2'/>
|
||||
<source file='/home/VMs/lq/openEuler-22.03-LTS-aarch64.qcow2'/>
|
||||
<target dev='vdc' bus='virtio'/>
|
||||
<address type='pci' domain='0x0000' bus='0x04' slot='0x00' function='0x0'/>
|
||||
</disk>
|
||||
<disk type='file' device='cdrom'>
|
||||
<driver name='qemu' type='raw'/>
|
||||
<target dev='sda' bus='scsi'/>
|
||||
<readonly/>
|
||||
<address type='drive' controller='0' bus='0' target='0' unit='0'/>
|
||||
</disk>
|
||||
<controller type='usb' index='0' model='qemu-xhci' ports='8'>
|
||||
<address type='pci' domain='0x0000' bus='0x02' slot='0x00' function='0x0'/>
|
||||
</controller>
|
||||
<controller type='scsi' index='0' model='virtio-scsi'>
|
||||
<address type='pci' domain='0x0000' bus='0x03' slot='0x00' function='0x0'/>
|
||||
</controller>
|
||||
<controller type='pci' index='0' model='pcie-root'/>
|
||||
<controller type='pci' index='1' model='pcie-root-port'>
|
||||
<model name='pcie-root-port'/>
|
||||
<target chassis='1' port='0x8'/>
|
||||
<address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x0' multifunction='on'/>
|
||||
</controller>
|
||||
<controller type='pci' index='2' model='pcie-root-port'>
|
||||
<model name='pcie-root-port'/>
|
||||
<target chassis='2' port='0x9'/>
|
||||
<address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x1'/>
|
||||
</controller>
|
||||
<controller type='pci' index='3' model='pcie-root-port'>
|
||||
<model name='pcie-root-port'/>
|
||||
<target chassis='3' port='0xa'/>
|
||||
<address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x2'/>
|
||||
</controller>
|
||||
<controller type='pci' index='4' model='pcie-root-port'>
|
||||
<model name='pcie-root-port'/>
|
||||
<target chassis='4' port='0xb'/>
|
||||
<address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x3'/>
|
||||
</controller>
|
||||
<controller type='pci' index='5' model='pcie-root-port'>
|
||||
<model name='pcie-root-port'/>
|
||||
<target chassis='5' port='0xc'/>
|
||||
<address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x4'/>
|
||||
</controller>
|
||||
<interface type='network'>
|
||||
<mac address='52:54:00:41:63:1e'/>
|
||||
<source network='default'/>
|
||||
<model type='virtio'/>
|
||||
<address type='pci' domain='0x0000' bus='0x01' slot='0x00' function='0x0'/>
|
||||
</interface>
|
||||
<serial type='pty'>
|
||||
<target type='system-serial' port='0'>
|
||||
<model name='pl011'/>
|
||||
</target>
|
||||
</serial>
|
||||
<console type='pty'>
|
||||
<target type='serial' port='0'/>
|
||||
</console>
|
||||
<memballoon model='virtio'>
|
||||
<stats period='10'/>
|
||||
<address type='pci' domain='0x0000' bus='0x00' slot='0x02' function='0x0'/>
|
||||
</memballoon>
|
||||
<graphics type="vnc" port="5992" autoport="yes" keymap="en-us" listen="127.0.0.3"/>
|
||||
<!--<hostdev mode='subsystem' type='pci' managed='yes'>
|
||||
<driver name='vfio'/>
|
||||
<source>
|
||||
<address domain='0x0000' bus='0x09' slot='0x00' function='0x0'/>
|
||||
</source>
|
||||
</hostdev>-->
|
||||
</devices>
|
||||
</domain>
|
Loading…
x
Reference in New Issue
Block a user