sylar高性能服务器-日志(P43-P48)内容记录

news2025/2/25 23:16:45

文章目录

    • P43:Hook01
      • 一、HOOK
        • 定义接口函数指针
        • 获取接口原始地址
      • 二、测试
    • P44-P48:Hook02-06
      • 一、hook实现基础
      • 二、class FdCtx
        • 成员变量
        • 构造函数
        • init
        • setTimeout
        • getTimeout
      • 三、class FdManager
        • 成员变量
        • 构造函数
        • get(获取/创建文件句柄类)
        • del(删除文件句柄类)
      • 四、class hook
        • do_io(socket操作真正执行体)
        • 定时器
        • do_io
        • sleep系列
        • Socket系列
        • connect系列
        • accept
        • close
        • ioctl(对设备进行控制操作)
      • 五、测试

P43:Hook01

sylar封装hook模块的目的就是让一些C标准库提供的同步API可以实现异步的性能。

hookAPI封装成一个与原始系统调用同名的接口,在调用这个接口时,先实现一些别的操作,然后在调用原始的系统API。这样对开发者来说很方便,不用重新学习新的接口,用着同步的接口实现异步的操作。

一、HOOK

将函数接口都存放到extern "C"作用域下,指定函数按照C语言的方式进行编译和链接。它的作用是为了解决C++中函数名重载的问题,使得C++代码可以和C语言代码进行互操作。

定义接口函数指针
extern "C" {

// 重新定义同名的接口

// sleep_fun 为函数指针
typedef unsigned int (*sleep_fun)(unsigned int seconds);
// 它是一个sleep_fun类型的函数指针变量,表示该变量在其他文件中已经定义,我们只是在当前文件中引用它。
extern sleep_fun sleep_f;

typedef int (*usleep_fun)(useconds_t usec);
extern usleep_fun usleep_f;

}

unsigned int sleep(unsigned int seconds) {
    if(!sylar::t_hook_enable) {
        return sleep_f(seconds);
    }

    sylar::Fiber::ptr fiber = sylar::Fiber::GetThis();
    sylar::IOManager* iom = sylar::IOManager::GetThis();
    iom->addTimer(seconds * 1000, std::bind((void(sylar::Scheduler::*)
            (sylar::Fiber::ptr, int thread))&sylar::IOManager::schedule
            ,iom, fiber, -1));
    sylar::Fiber::YieldToHold();
    return 0;

}

int usleep(useconds_t usec) {
    if(!sylar::t_hook_enable) {
        return usleep_f(usec);
    }

    sylar::Fiber::ptr fiber = sylar::Fiber::GetThis();
    sylar::IOManager* iom = sylar::IOManager::GetThis();
    iom->addTimer(usec / 1000, std::bind((void(sylar::Scheduler::*)
            (sylar::Fiber::ptr, int thread))&sylar::IOManager::schedule
            ,iom, fiber, -1));
    sylar::Fiber::YieldToHold();
    return 0;
}


获取接口原始地址

使用宏来封装对每个原始接口地址的获取。

#define HOOK_FUN(XX) \
    XX(sleep) \
    XX(usleep)

hook_init()封装到一个结构体的构造函数中,并创建静态对象,能够在main函数运行之前就能将地址保存到函数指针变量当中。

void hook_init() {
    static bool is_inited = false;
    if(is_inited) {
        return;
    }
// dlsym - 从一个动态链接库或者可执行文件中获取到符号地址。成功返回跟name关联的地址
// RTLD_NEXT 返回第一个匹配到的 "name" 的函数地址
// 取出原函数,赋值给新函数
#define XX(name) name ## _f = (name ## _fun)dlsym(RTLD_NEXT, #name);
    HOOK_FUN(XX);
#undef XX
}

struct _HookIniter {
    _HookIniter() {
        hook_init();
    }
};

static _HookIniter s_hook_initer;

extern "C" {
// 声明变量
#define XX(name) name ## _fun name ## _f = nullptr;
    HOOK_FUN(XX);
#undef XX
}

上面的宏展开如下

extern "C" {
	sleep_fun sleep_f = nullptr;
    usleep_fun usleep_f = nullptr;
}

void hook_init() {
    static bool is_inited = false;
    if (is_inited) {
        return;
    }
    
	sleep_f = (sleep_fun)dlsym(RTLD_NEXT, "sleep");
    usleep_f = (usleep_fun)dlsym(RTLD_NEXT, "usleep");

二、测试

#include"../sylar/hook.h"
#include"../sylar/log.h"
#include"../sylar/iomanager.h"

sylar::Logger::ptr g_logger = SYLAR_LOG_ROOT();

void test_sleep() {
    SYLAR_LOG_INFO(g_logger) << "start()_HY";
    sylar::IOManager iom(1);
    iom.schedule([](){
        sleep(2);
        SYLAR_LOG_INFO(g_logger) << "sleep 2";
    });

    iom.schedule([](){
        sleep(3);
        SYLAR_LOG_INFO(g_logger) << "sleep 3";
    });
    SYLAR_LOG_INFO(g_logger) << "test_sleep";
}

int main(int argc, char** argv) {
    test_sleep();
    return 0;
}

上面测试用例通过IOManager进行两次调度,第一个任务睡眠2s,第二个任务睡眠3s

不开启hook

在不开启hook的情况下,两个任务必须按着顺序执行,也就是一共需要5s时间

image-20240226193319313

开启hook

在执行hook情况下,第一个任务在执行sleep时就添加一个2s的定时器,利用回调函数去调度本协程,然后让协程让出执行权去处理第二个任务,这样完成两个任务一共只需要3s,通过hook让只能同步的sleep实现了异步的效果

unsigned int sleep(unsigned int seconds) {
    if(!sylar::t_hook_enable) {
        return sleep_f(seconds);
    }

    sylar::Fiber::ptr fiber = sylar::Fiber::GetThis();
    sylar::IOManager* iom = sylar::IOManager::GetThis();
    iom->addTimer(seconds * 1000, std::bind((void(sylar::Scheduler::*)
            (sylar::Fiber::ptr, int thread))&sylar::IOManager::schedule
            ,iom, fiber, -1));
    sylar::Fiber::YieldToHold();
    return 0;

}

image-20240226193645237

P44-P48:Hook02-06

以下记录参考博客链接

一、hook实现基础

​ 通过动态库的全局符号介入,使用自定义的接口替换掉同名的系统调用接口。

​ 系统调用接口基本上是由C标准函数库libc提供,所以这里要做的事情就是用自定义的动态库来覆盖掉libc中的同名符号。

​ 由于动态库的全局符号介入问题,全局符号表只会记录第一次识别到的符号,后续的同名符号都被忽略,但这并不表示同名符号所在的动态库完全不会加载,因为有可能其他的符号会用到。

​ 以libc库举例,如果用户在链接libc库之前链接了一个指定的库,并且在这个库里实现了read/write接口,那么在程序运行时,程序调用的read/write接口就是指定库里的,而不是libc库里的。libc库仍然会被加载,因为libc库是程序的运行时库,程序不可能不依赖libc里的其他接口。因为libc库也被加载了,所以,通过一定的手段,仍然可以从libc中拿到属于libcread/write接口,这就为hook创建了条件。程序可以定义自己的read/write接口,在接口内部先实现一些相关的操作,然后再调用libc里的read/write接口。(写得真清楚!)

而将libc库中的接口重新找回来的方法就是使用dlsym()

#include <dlfcn.h>

/*
 * 第一个参数固定为 RTLD_NEXT,第二个参数为符号的名称
 */
void *dlsym(void *handle, const char *symbol);

二、class FdCtx

FdCtx存储每一个fd相关的信息,并由FdManager管理每一个FdCtxFdManager为单例类

成员变量
// 是否初始化
bool m_isInit: 1;
// 是否Socket
bool m_isSocket: 1;
// 是否hook非阻塞
bool m_sysNonblock: 1;
// 是否用户主动设置非阻塞
bool m_userNonblock: 1;
// 是否关闭
bool m_isClosed: 1;
// 文件句柄
int m_fd;
// 读超时时间毫秒
uint64_t m_recvTimeout;
// 写超时时间毫秒
uint64_t m_sendTimeout;
sylar::IOManager* m_iomanager;
构造函数
// 构造函数
FdCtx::FdCtx(int fd)
    : m_isInit(false)
    , m_isSocket(false)
    , m_sysNonblock(false)
    , m_userNonblock(false)
    , m_isClosed(false)
    , m_fd(fd)
    , m_recvTimeout(-1)
    , m_sendTimeout(-1) {
    init();
}
init
// 初始化
bool FdCtx::init() {
    // 如果已经初始化直接返回true
    if(m_isInit) {
        return true;
    }
    // 默认发送/接收超时时间
    m_recvTimeout = -1;
    m_sendTimeout = -1;

    // 定义一个stat结构体
    struct stat fd_stat;
    // 通过文件描述符取得文件的状态,返回-1失败
    if(-1 == fstat(m_fd, &fd_stat)) {
        // 初始化失败并且不是Socket
        m_isInit = false;
        m_isSocket = false;
    } else {
        // 初始化成功
        m_isInit = true;
        // S_ISSOCK (st_mode) 是否为socket 
        m_isSocket = S_ISSOCK(fd_stat.st_mode);
    }

    // 如果是socket,则给它设置为阻塞状态
    if(m_isSocket) {
        // 获取文件的flags
        int flags = fcntl_f(m_fd, F_GETFL, 0);
        // 判断是否是阻塞的
        if(!(flags & O_NONBLOCK)) {
            // 不是则设置为阻塞
            fcntl_f(m_fd, F_SETFL, flags | O_NONBLOCK);
        }
        // 阻塞设置生效
        m_sysNonblock = true;
    } else {
        // 不是Socket则不管
        m_sysNonblock = false;
    }
    // 初始化用户没有设置为阻塞
    m_userNonblock = false;
    // 未关闭
    m_isClosed = false;
    // 反正初始化状态
    return m_isInit;
}

strcut stat

struct stat    
{    
    dev_t       st_dev;     /* ID of device containing file -文件所在设备的ID*/    
    ino_t       st_ino;     /* inode number -inode节点号*/  
    mode_t      st_mode;    /* 文件的类型和存取的权限*/    
    nlink_t     st_nlink;   /* number of hard links -链向此文件的连接数(硬连接)*/    
    uid_t       st_uid;     /* user ID of owner -user id*/    
    gid_t       st_gid;     /* group ID of owner - group id*/    
    dev_t       st_rdev;    /* device ID (if special file) -设备号,针对设备文件*/    
    off_t       st_size;    /* total size, in bytes -文件大小,字节为单位*/    
    blksize_t   st_blksize; /* blocksize for filesystem I/O -系统块的大小*/    
    blkcnt_t    st_blocks;  /* number of blocks allocated -文件所占块数*/    
    time_t      st_atime;   /* time of last access -最近存取时间*/    
    time_t      st_mtime;   /* time of last modification -最近修改时间*/    
    time_t      st_ctime;   /* time of last status change - */    
};    

fstat

通过文件描述符取得文件的状态

int fstat (int __fd, struct stat *__buf)
setTimeout
// 设置超时时间
void FdCtx::setTimeout(int type, uint64_t v) {
    // 套接字为设置Socket接收数据的超时时间
    if(type == SO_RCVTIMEO) {
        m_recvTimeout = v;
    } else {
        m_sendTimeout = v;
    }
}
getTimeout
// 获取超时时间
uint64_t FdCtx::getTimeout(int type) {
    if(type == SO_RCVTIMEO) {
        return m_recvTimeout;
    } else {
        return m_sendTimeout;
    }
}

三、class FdManager

成员变量
// 读写锁
RWMutexType m_mutex;
// 文件句柄集合
std::vector<FdCtx::ptr> m_datas;
构造函数
// 构造函数
FdManager::FdManager() {
    m_datas.resize(64);
}
get(获取/创建文件句柄类)
// 获取/创建文件句柄类
FdCtx::ptr FdManager::get(int fd, bool auto_create) {
    RWMutexType::ReadLock lock(m_mutex);
    // 表示集合中没有,并且也不自动创建,直接返回空指针
    if((int)m_datas.size() <= fd) {
        if(auto_create ==false) {
            return nullptr;
        }
    } else {
        // 当前有值或者不需要创建,直接返回目标值
        if(m_datas[fd] || !auto_create) {
            return m_datas[fd];
        }
    }
    lock.unlock();
    // 自动创建
    RWMutexType::WriteLock lock2(m_mutex);
    FdCtx::ptr ctx(new FdCtx(fd));
    m_datas[fd] = ctx;
    return ctx;
}
del(删除文件句柄类)
// 删除文件句柄类
void FdManager::del(int fd) {
    RWMutexType::WriteLock lock(m_mutex);
    // 没找到直接返回
    if((int)m_datas.size() <= fd) {
        return;
    }
    // 删除
    m_datas[fd].reset();
}

四、class hook

这里补充一些函数方法

do_io(socket操作真正执行体)

流程

  • 先进行一系列判断,是否按原函数执行。

  • 执行原始函数进行操作,若errno = EINTR,则为系统中断,应该不断重新尝试操作。

  • errno = EAGIN,系统已经隐式的将socket设置为非阻塞模式,此时资源咱不可用。

  • 若设置了超时时间,则设置一个执行周期为超时时间的条件定时器,它保证若在超时之前数据就已经来了,然后操作完do_io执行完毕,智能指针tinfo已经销毁了,但是定时器还在,此时弱指针拿到的就是个空指针,将不会执行定时器的回调函数。

  • 在条件定时器的回调函数中设置错误为ETIMEDOUT超时,并且使用cancelEvent强制执行该任务,继续回到该协程执行。

  • 通过addEvent添加事件,若添加事件失败,则将条件定时器删除并返回错误。成功则让出协程执行权。

  • 只有两种情况协程会被拉起: a. 超时了,通过定时器回调函数 cancelEvent ---> triggerEvent会唤醒回来 b. addEvent数据回来了会唤醒回来

  • 将定时器取消,若为超时则返回-1并设置errno = ETIMEDOUT,并返回-1。

  • 若为数据来了则retry,重新操作。

定时器
// 定时器超时条件
struct timer_info {
    int cancelled = 0;
};
do_io

/*

* fd 文件描述符

* fun 原始函数

* hook_fun_name hook的函数名称

* event 事件

* timeout_so 超时时间类型

* args 可变参数

*

* 例如:return do_io(fd, read_f, “read”, sylar::IOManager::READ, SO_RCVTIMEO, buf, count);

*/

template<typename OriginFun, typename... Args>
static ssize_t do_io(int fd, OriginFun fun, const char* hook_fun_name,
        uint32_t event, int timeout_so, Args&&... args) {
    // 非hook直接返回原接口
    if(!sylar::t_hook_enable) {
        /* 可以将传入的可变参数args以原始类型的方式传递给函数fun。
         * 这样做的好处是可以避免不必要的类型转换和拷贝,提高代码的效率和性能。*/
        return fun(fd, std::forward<Args>(args)...);
    }
    // 通过文件句柄获得对应的FdCtx
    sylar::FdCtx::ptr ctx = sylar::FdMgr::GetInstance()->get(fd);
    // 没有文件调原接口
    if(!ctx) {
        return fun(fd, std::forward<Args>(args)...);
    }
    // 如果句柄已经关闭
    if(ctx->isClose()) {
        // #define	EBADF		 9	/* Bad file number */
        errno = EBADF;
        return -1;
    }
    // 如果不是Socket或者用户设置了非阻塞,仍然调原接口
    if(!ctx->isSocket() || ctx->getUserNonblock()) {
        return fun(fd, std::forward<Args>(args)...);
    }
    // ------ hook要做了 ------异步IO
    // 获得超时时间
    uint64_t to = ctx->getTimeout(timeout_so);
    // 设置超时条件
    std::shared_ptr<timer_info> tinfo(new timer_info);

retry:
    // 先执行fun 读数据或写数据 若函数返回值有效就直接返回
    // std::forward 是一个 C++11 中的模板函数,其主要作用是在模板函数或模板类中,
    // 将一个参数以“原样”(forward)的方式转发给另一个函数
    ssize_t n = fun(fd, std::forward<Args>(args)...);
    // 若中断则重试
    // #define	EINTR		 4	/* Interrupted system call */
    while(n == -1 && errno == EINTR) {
        n = fun(fd, std::forward<Args>(args)...);
    }
    // 重试
    // #define	EAGAIN		11	/* Try again */
    if(n == -1 && errno == EAGAIN) {
        // 获得当前IO调度器
        sylar::IOManager* iom = sylar::IOManager::GetThis();
        // 定时器
        sylar::Timer::ptr timer;
        // tinfo的弱指针,可以判断tinfo是否已经销毁
        std::weak_ptr<timer_info> winfo(tinfo);
        // 设置了超时时间
        if(to != (uint64_t)-1) {
            // 添加条件定时器
            // to时间到了消息还没来就触发callback
            timer = iom->addConditionTimer(to, [winfo, fd, iom, event]() {
                auto t = winfo.lock();
                // 定时器失效
                if(!t || t->cancelled) {
                    return;
                }
                // 没错误的话设置为超时而失败
                // #define	ETIMEDOUT	110	/* Connection timed out */
                t->cancelled = ETIMEDOUT;
                // 取消事件强制唤醒
                iom->cancelEvent(fd, (sylar::IOManager::Event)(event));
            }, winfo);
        }
        // 默认cb为空,任务执行当前协程
        int rt = iom->addEvent(fd, (sylar::IOManager::Event)(event));
        // addEvent失败, 取消上面加的定时器
        if(rt) {
            SYLAR_LOG_ERROR(g_logger) << hook_fun_name << " addEvent("
                << fd << ", " << event << ")";
            if(timer) {
                timer->cancel();
            }
            return -1;
        } else {
            /*	addEvent成功,把执行时间让出来
             *	只有两种情况会从这回来:
             * 	1) 超时了, timer cancelEvent triggerEvent会唤醒回来
             * 	2) addEvent数据回来了会唤醒回来 
            */
            SYLAR_LOG_DEBUG(g_logger) << "do_io <" << hook_fun_name << ">";
            sylar::Fiber::YieldToHold();
            SYLAR_LOG_DEBUG(g_logger) << "do_io <" << hook_fun_name << ">";
            if(timer) {
                timer->cancel();
            }
            // 从定时任务唤醒,超时失败
            if(tinfo->cancelled) {
                errno = tinfo->cancelled;
                return -1;
            }
            // 数据来了就直接重新去操作
            goto retry;
        }
    }
    return n;
}
sleep系列

设置一个定时器然后让出执行权,超时后继续执行该协程。

回调函数使用std::bind函数将sylar::IOManager::schedule函数绑定到iom对象上,并传入fiber-1两个参数。由于schedule是个模板类,如果直接与函数绑定,就无法确定函数的类型,从而无法使用std::bind函数。因此,需要先声明函数指针,将函数的类型确定下来,然后再将函数指针与std::bind函数进行绑定。

unsigned int sleep(unsigned int seconds) {
    if(!sylar::t_hook_enable) {
        return sleep_f(seconds);
    }

    sylar::Fiber::ptr fiber = sylar::Fiber::GetThis();
    sylar::IOManager* iom = sylar::IOManager::GetThis();
     /**
     * 	@details
     *
     *	(void(sylar::Scheduler::*)(sylar::Fiber::ptr, int thread)) 是一个函数指针类型,
     *	它定义了一个指向 sylar::Scheduler 类中一个参数为 sylar::Fiber::ptr 和 int 类型的成员函数的指针类型。
     *	具体来说,它的含义如下:
     *	void 表示该成员函数的返回值类型,这里是 void 类型。
     *	(sylar::Scheduler::*) 表示这是一个 sylar::Scheduler 类的成员函数指针类型。
     *	(sylar::Fiber::ptr, int thread) 表示该成员函数的参数列表
     *       ,其中第一个参数为 sylar::Fiber::ptr 类型,第二个参数为 int 类型。
     *	
     *	使用 std::bind 绑定了 sylar::IOManager::schedule 函数,
     * 	并将 iom 实例作为第一个参数传递给了 std::bind 函数,将sylar::IOManager::schedule函数绑定到iom对象上。
     * 	在这里,第二个参数使用了函数指针类型 (void(sylar::Scheduler::*)(sylar::Fiber::ptr, int thread))
     * 	,表示要绑定的函数类型是 sylar::Scheduler 类中一个参数为 sylar::Fiber::ptr 和 int 类型的成员函数
     * 	,这样 std::bind 就可以根据这个函数类型来实例化出一个特定的函数对象,并将 fiber 和 -1 作为参数传递给它。
     */

    iom->addTimer(seconds * 1000, std::bind((void(sylar::Scheduler::*)
            (sylar::Fiber::ptr, int thread))&sylar::IOManager::schedule
            ,iom, fiber, -1));
    sylar::Fiber::YieldToHold();
    return 0;
}
Socket系列
int socket(int domain, int type, int protocol) {
    if(!sylar::t_hook_enable) {
        return socket_f(domain, type, protocol);
    }
    int fd = socket_f(domain, type, protocol);
    if(fd == -1) {
        return fd;
    }
    // 将fd放入到文件管理中
    sylar::FdMgr::GetInstance()->get(fd, true);
    return fd;
}
connect系列
int connect_with_timeout(int fd, const struct sockaddr* addr, socklen_t addrlen, uint64_t timeout_ms) {
    if(!sylar::t_hook_enable) {
        return connect_f(fd, addr, addrlen);
    }
    sylar::FdCtx::ptr ctx = sylar::FdMgr::GetInstance()->get(fd);
    if(!ctx || ctx->isClose()) {
        errno = EBADF;
        return -1;
    }

    if(!ctx->isSocket()) {
        return connect_f(fd, addr, addrlen);
    }

    if(ctx->getUserNonblock()) {
        return connect_f(fd, addr, addrlen);
    }
    // 异步开始
    // 尝试连接
    int n = connect_f(fd, addr, addrlen);
    // 连接成功
    if(n == 0) {
        return 0;
    // 失败 #define	EINPROGRESS	115	/* Operation now in progress */
    } else if(n != -1 || errno != EINPROGRESS) {
        return n;
    }

    sylar::IOManager* iom = sylar::IOManager::GetThis();
    sylar::Timer::ptr timer;
    std::shared_ptr<timer_info> tinfo(new timer_info);
    std::weak_ptr<timer_info> winfo(tinfo);

    // 设置了超时时间
    if(timeout_ms != (uint64_t)-1) {
        // 加条件定时器
        timer = iom->addConditionTimer(timeout_ms, [winfo, fd, iom]() {
                auto t = winfo.lock();
                if(!t || t->cancelled) {
                    return;
                }
                t->cancelled = ETIMEDOUT;
                iom->cancelEvent(fd, sylar::IOManager::WRITE);
        }, winfo);
    }
    // 添加一个写事件
    int rt = iom->addEvent(fd, sylar::IOManager::WRITE);
    if(rt == 0) {
        /* 	只有两种情况唤醒:
         * 	1. 超时,从定时器唤醒
         *	2. 连接成功,从epoll_wait拿到事件 */
        sylar::Fiber::YieldToHold();
        if(timer) {
            timer->cancel();
        }
        // 从定时器唤醒,超时失败
        if(tinfo->cancelled) {
            errno = tinfo->cancelled;
            return -1;
        }
    } else {
        // 添加事件失败
        if(timer) {
            timer->cancel();
        }
        SYLAR_LOG_ERROR(g_logger) << "connect addEvent(" << fd << ", WRITE) error";
    }

    int error = 0;
    socklen_t len = sizeof(int);
    // 获取套接字的错误状态
    if(-1 == getsockopt(fd, SOL_SOCKET, SO_ERROR, &error, &len)) {
        return -1;
    }
    // 没有错误,连接成功
    if(!error) {
        return 0;
    } else {
        errno = error;
        return -1;
    }
}
accept
int accept(int s, struct sockaddr *addr, socklen_t *addrlen) {
    int fd = do_io(s, accept_f, "accept", sylar::IOManager::READ, SO_RCVTIMEO, addr, addrlen);
    // 将新创建的连接放到文件管理中
    if(fd >= 0) {
        sylar::FdMgr::GetInstance()->get(fd, true);
    }
    return fd;
}
close
int close(int fd) {
    if(!sylar::t_hook_enable) {
        return close_f(fd);
    }

    sylar::FdCtx::ptr ctx = sylar::FdMgr::GetInstance()->get(fd);
    if(ctx) {
        auto iom = sylar::IOManager::GetThis();
        // 取消事件
        if(iom) {
            iom->cancelAll(fd);
        }
        // 在文件管理中删除
        sylar::FdMgr::GetInstance()->del(fd);
    }
    return close_f(fd);
}

fcntl(修改文件状态)

int fcntl(int fd, int cmd, ... /* arg */ ) {
    va_list va;
    va_start(va, cmd);
    switch(cmd) {
        case F_SETFL:
            {
                int arg = va_arg(va, int);
                va_end(va);
                sylar::FdCtx::ptr ctx = sylar::FdMgr::GetInstance()->get(fd);
                if(!ctx || ctx->isClose() || !ctx->isSocket()) {
                    return fcntl_f(fd, cmd, arg);
                }
                ctx->setUserNonblock(arg & O_NONBLOCK);
                if(ctx->getSysNonblock()) {
                    arg |= O_NONBLOCK;
                } else {
                    arg &= ~O_NONBLOCK;
                }
                return fcntl_f(fd, cmd, arg);
            }
            break;
        case F_GETFL:
            {
                va_end(va);
                int arg = fcntl_f(fd, cmd);
                sylar::FdCtx::ptr ctx = sylar::FdMgr::GetInstance()->get(fd);
                if(!ctx || ctx->isClose() || !ctx->isSocket()) {
                    return arg;
                }
                if(ctx->getUserNonblock()) {
                    return arg | O_NONBLOCK;
                } else {
                    return arg & ~O_NONBLOCK;
                }
            }
            break;
        case F_DUPFD:
        case F_DUPFD_CLOEXEC:
        case F_SETFD:
        case F_SETOWN:
        case F_SETSIG:
        case F_SETLEASE:
        case F_NOTIFY:
#ifdef F_SETPIPE_SZ
        case F_SETPIPE_SZ:
#endif
            {
                int arg = va_arg(va, int);
                va_end(va);
                return fcntl_f(fd, cmd, arg); 
            }
            break;
        case F_GETFD:
        case F_GETOWN:
        case F_GETSIG:
        case F_GETLEASE:
#ifdef F_GETPIPE_SZ
        case F_GETPIPE_SZ:
#endif
            {
                va_end(va);
                return fcntl_f(fd, cmd);
            }
            break;
        case F_SETLK:
        case F_SETLKW:
        case F_GETLK:
            {
                struct flock* arg = va_arg(va, struct flock*);
                va_end(va);
                return fcntl_f(fd, cmd, arg);
            }
            break;
        case F_GETOWN_EX:
        case F_SETOWN_EX:
            {
                struct f_owner_exlock* arg = va_arg(va, struct f_owner_exlock*);
                va_end(va);
                return fcntl_f(fd, cmd, arg);
            }
            break;
        default:
            va_end(va);
            return fcntl_f(fd, cmd);
    }
}
ioctl(对设备进行控制操作)
int ioctl(int d, unsigned long int request, ...) {
    va_list va;
    va_start(va, request);
    void* arg = va_arg(va, void*);
    va_end(va);
    //	FIONBIO用于设置文件描述符的非阻塞模式
    if(FIONBIO == request) {
        bool user_nonblock = !!*(int*)arg;
        sylar::FdCtx::ptr ctx = sylar::FdMgr::GetInstance()->get(d);
        if(!ctx || ctx->isClose() || !ctx->isSocket()) {
            return ioctl_f(d, request, arg);
        }
        ctx->setUserNonblock(user_nonblock);
    }
    return ioctl_f(d, request, arg);
}

setsockopt(设置Socket)

int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen) {
    if(!sylar::t_hook_enable) {
        return setsockopt_f(sockfd, level, optname, optval, optlen);
    }
    // 如果设置socket通用选项
    if(level == SOL_SOCKET) {
         // 如果设置超时选项
        if(optname == SO_RCVTIMEO || optname == SO_SNDTIMEO) {
            sylar::FdCtx::ptr ctx = sylar::FdMgr::GetInstance()->get(sockfd);
            if(ctx) {
                const timeval* v = (const timeval*)optval;
                // 转为毫秒保存
                ctx->setTimeout(optname, v->tv_sec * 1000 + v->tv_usec / 1000);
            }
        }
    }
    return setsockopt_f(sockfd, level, optname, optval, optlen);
}

五、测试

代码

    int sock = socket(AF_INET, SOCK_STREAM, 0);
    sockaddr_in addr;
    memset(&addr, 0, sizeof(addr));
    addr.sin_family = AF_INET;
    addr.sin_port = htons(80);
    inet_pton(AF_INET, "183.2.172.185", &addr.sin_addr.s_addr);
    SYLAR_LOG_INFO(g_logger) << "begin connect";
    int rt = connect(sock, (const sockaddr*)&addr, sizeof(addr));
    SYLAR_LOG_INFO(g_logger) << "connect rt = " << " errno = " << errno;
    if(rt) {
        return;
    }
    // 发送消息
    const char data[] = "GET / HTTP/1.0\r\n\r\n";
    rt = send(sock,data,sizeof(data), 0);
    SYLAR_LOG_INFO(g_logger) << "send rt = " << rt << " error = " << errno;
    if(rt <= 0) {
        return;
    }

    // 接收消息
    std::string buff;
    buff.resize(4096);

    rt = recv(sock, &buff[0], buff.size(), 0);
    SYLAR_LOG_INFO(g_logger) << "recv rt = " << rt << " errno = " << errno;

    if(rt <= 0) {
        return;
    }
    buff.resize(rt);
    SYLAR_LOG_INFO(g_logger) << buff;

}

结果

image-20240228165004953

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1477569.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

前端架构: 脚手架之包管理工具的案例对比及workspaces特性的使用与发布过程

npm的workspaces 特性 1 &#xff09;使用或不使用包管理工具的对比 vue-cli 这个脚手架使用 Lerna 管理&#xff0c;它的项目显得非常清晰在 vue-cli 中包含很多 package 点开进去&#xff0c;每一个包都有package.json它里面有很多项目&#xff0c;再没有 Lerna 之前去维护和…

【Spring】回顾反射机制

一、分析方法四要素 package org.qiu.reflect;/*** author 秋玄* version 1.0* email qiu_2022aliyun.com* project Spring* package org.qiu.reflect* date 2022-11-11-17:26* since 1.0*/ public class SomeService {public void doSome(){System.out.println("public …

【Unity】如何在Unity 中创建带有缩放效果的滚动视图(具有吸附效果的实现与优化)?

效果预览&#xff1a; 目录 效果预览&#xff1a; 一、引言&#xff1a; 二、问题描述 三、解决方案&#xff1a; 三、优化&#xff1a; 四、结论 一、引言&#xff1a; 在Unity开发中&#xff0c;经常需要实现滚动视图&#xff08;ScrollView&#xff09;中的内容吸附到…

个人博客系列-Django部署-nginx(3)

使用Nginx uwsgi进行部署django项目 一. 检查项目是否可以运行 启动项目 python manage.py runserver 0.0.0.0:8099输入ip:8099 查看启动页面 出现上述页面表示运行成功 二. 安装uwsgi并配置 2.1 下载uwsgi pip install uwsgi新建文件test.py写入内容&#xff0c;测试一…

面试笔记系列六之redis+kafka+zookeeper基础知识点整理及常见面试题

目录 Redis redis持久化机制&#xff1a;RDB和AOF Redis 持久化 RDB的优点 RDB的缺点 AOF 优点 AOF 缺点 4.X版本的整合策略 Redis做分布式锁用什么命令&#xff1f; Redis做分布式锁死锁有哪些情况&#xff0c;如何解决&#xff1f; Redis如何做分布式锁&#xff1f…

Logic Pro:专业音乐制作软件,为你的音乐插上翅膀

Logic Pro是一款功能强大的音乐制作软件&#xff0c;专为专业音乐人和音乐爱好者设计。它提供了全面的音乐创作工具&#xff0c;包括音频录音、编辑、混音、合成以及自动化等功能&#xff0c;让你能够轻松实现音乐梦想。 Logic Pro软件获取 首先&#xff0c;Logic Pro拥有卓越…

Tomcat:Java Web

简介 Apache Tomcat 是 Java Web 应用程序开发中最为常用的服务器之一。作为一个开源、轻量级的 Servlet 容器和 JSP 容器&#xff0c;Tomcat 提供了一个稳定可靠的运行环境&#xff0c;使得开发者可以快速开发、部署和管理 Java Web 应用程序。本文将深入介绍 Tomcat 的特点、…

js 面试 sort背后原理及排序使用方法

1 sort 概念 2 原理 之前版本是用的插入排序 和 快速排序。 现在用的是冒泡排序。 3 排序方式&#xff1a; 2-1 根据字符码点排序 2-2 从小到大或从大到小排序 2-3 对数组中的对象进行排序

高校水电预付费管控系统

高校水电预付费管控系统在现代高校管理中扮演着重要角色。这一系统通过整合先进的科技手段&#xff0c;如智能计量设备和互联网技术&#xff0c;实现对校园水电消费的精准监控和管理。首先&#xff0c;高校水电预付费管控系统能够有效监测学生宿舍、教学楼等区域的实时用水用电…

Zookeeper基础入门-2【ZooKeeper 分布式锁案例】

Zookeeper基础入门-2【ZooKeeper 分布式锁案例】 四、ZooKeeper-IDEA环境搭建4.1.环境搭建4.1.1.创建maven工程&#xff1a;zookeeper4.1.2.在pom文件添加依赖4.1.3.在项目的src/main/resources 目录下&#xff0c;新建文件为“log4j.properties”4.1.4.创建包名com.orange.zk …

网站数据加密之Hook通用方案

文章目录 1. 写在前面2. 请求分析3. 编写Hook4. 其他案例 【作者主页】&#xff1a;吴秋霖 【作者介绍】&#xff1a;Python领域优质创作者、阿里云博客专家、华为云享专家。长期致力于Python与爬虫领域研究与开发工作&#xff01; 【作者推荐】&#xff1a;对JS逆向感兴趣的朋…

Python进阶学习:Pickle模块--dump()和load()的用法

Python进阶学习&#xff1a;Pickle模块–dump()和load()的用法 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f448; 希望得到您…

【c语言】探索联合和枚举---解锁更多选择

前言 上一篇 讲解的是结构体相关知识&#xff0c;接着本篇主要讲解的是 联合和枚举 相关知识 结构体、联合体和枚举都属于 自定义类型。 那么接下来就跟上我的节奏&#xff0c;准备发车~ 欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xf…

网络协议:DHCP协议工作原理,DHCP分配方式,DHCP租约,Wireshark抓包分析DHCP报文

「作者主页」&#xff1a;士别三日wyx 「作者简介」&#xff1a;CSDN top100、阿里云博客专家、华为云享专家、网络安全领域优质创作者 「专栏简介」&#xff1a;此文章已录入专栏《计算机网络零基础快速入门》 DHCP协议 一、简介二、分配方式1&#xff09;自动分配2&#xff0…

C++笔记(六)--- 静态成员变量/函数(static)

目录 C语言中静态变量 C静态成员变量 C静态成员函数 C语言中静态变量 1.函数内部用static修饰的变量&#xff0c;只能在此函数使用&#xff0c;但所修饰的变量不随函数释放而释放&#xff0c;下次调用时的值仍为上次结束时的值2.static修饰的全局变量只能在其定义的文件使用…

DAY9-防病毒AV概述

DNS过滤 URL过滤和DNS过滤对比

clickhouse 大表数据归档处理解决办法

本文作者为 360 奇舞团前端开发工程师 李彬 一、需求场景 基于目前项目在各个应用上的数据采集&#xff0c;尤其是性能监控上的采集&#xff0c;数据量越来越大&#xff0c;例如pv数据的采集表三个月的数据量已经在3亿&#xff0c;数据量小的也有几百万&#xff0c;但是目前的常…

浅谈 Linux fork 函数

文章目录 前言fork 基本概念代码演示示例1&#xff1a;体会 fork 函数返回值的作用示例2&#xff1a;创建多进程&#xff0c;加深对 fork 函数的理解 前言 本篇介绍 fork 函数。 fork 基本概念 pid_t fork(void) fork 的英文含义是"分叉"&#xff0c;在这里就是 …

TypeScript+React Web应用开发实战

&#x1f482; 个人网站:【 海拥】【神级代码资源网站】【办公神器】&#x1f91f; 基于Web端打造的&#xff1a;&#x1f449;轻量化工具创作平台&#x1f485; 想寻找共同学习交流的小伙伴&#xff0c;请点击【全栈技术交流群】 在现代Web开发中&#xff0c;React和TypeScrip…

嵌入式学习day25 Linux

进程基本概念: 1.进程: 程序&#xff1a;存放在外存中的一段数据组成的文件 进程&#xff1a;是一个程序动态执行的过程,包括进程的创建、进程的调度、进程的消亡 2.进程相关命令: 1.top 动态查看当前系统中的所有进程信息&#xff08;根据CPU占用率排序&a…