网络模块初始化
snmpd进程可以根据用户的配置来决定使用哪种协议来创建server,以某个协议为例子查看到接口注册流程如下:
init_snmp
_init_snmp
netsnmp_tdomain_init
snmp_transport_inits.h
netsnmp_tdomain_register(add domain_list(核心数据结构))
//snmp_transport_inits.h
//使用宏来区分使用udp还是tcp协议,如下:
#ifdef NETSNMP_TRANSPORT_UDP_DOMAIN //为1
netsnmp_udp_ctor();
#endif
#ifdef NETSNMP_TRANSPORT_TCP_DOMAIN //为1
netsnmp_tcp_ctor();
#endif
#ifdef NETSNMP_TRANSPORT_ALIAS_DOMAIN
netsnmp_alias_ctor();
#endif
#ifdef NETSNMP_TRANSPORT_UDPIPV6_DOMAIN
netsnmp_udpipv6_ctor();
#endif
#ifdef NETSNMP_TRANSPORT_TCPIPV6_DOMAIN
netsnmp_tcpipv6_ctor();
#endif
根据用户配置进行初始化流程如下:
init_master_agent
netsnmp_tdomain_transport_full
match = find_tdomain(mystring);//mystring即为用户的配置
match->f_create_from_tstring_new(执行注册好的回调函数创建server)
调用过程可以使用如此命令进行查看,程序执行过程中会将相应的token打印在终端上,方便梳理逻辑:
snmpd -Lo -f -Dsnmp_agent -Dread_config -Dtdomain
结果大致如下(可通过关键字查找代码逻辑):
···
snmp_agent: final port spec: "tcp:1601"
snmp_agent: installing master agent on port tcp:1601
tdomain: tdomain_transport_full("snmp", "tcp:1601", 1, "[NIL]", "[NIL]")
tdomain: Found domain "tcp" from specifier "tcp"
tdomain: trying domain "tcp" address "1601" default address ":161"
netsnmp_sockaddr_in: addr 0xffffc12f3a20, inpeername "1601", default_target ":161"
netsnmp_sockaddr_in: addr 0xffffc12f3a20, inpeername ":161", default_target "[NIL]"
netsnmp_sockaddr_in: check user service 161
netsnmp_sockaddr_in: return { AF_INET, 0.0.0.0:161 }
netsnmp_sockaddr_in: check user service 1601
netsnmp_sockaddr_in: return { AF_INET, 0.0.0.0:1601 }
···
关键数据结构与接口
typedef struct netsnmp_tdomain_s {
const oid *name;
size_t name_length;
const char **prefix;
/*
* The f_create_from_tstring field is deprecated, please do not use it
* for new code and try to migrate old code away from using it.
*/
netsnmp_transport *(*f_create_from_tstring) (const char *, int);
netsnmp_transport *(*f_create_from_ostring) (const u_char *, size_t, int);
struct netsnmp_tdomain_s *next;
netsnmp_transport *(*f_create_from_tstring_new) (const char *, int,
const char*);
} netsnmp_tdomain;
/*
* Our list of supported transport domains.
*/
static netsnmp_tdomain *domain_list = NULL;
int netsnmp_tdomain_register(netsnmp_tdomain *domain);
int netsnmp_tdomain_unregister(netsnmp_tdomain *domain);
static netsnmp_tdomain *find_tdomain(const char* spec);
实现原理
将底层的tcp与udp处理数据的接口封装好,在上层创建一个中间层transport来隐藏底层的具体实现,使用一个链表来将具体的实现作为子节点连接起来,利用反射的方式根据用户的配置去找到对应的接口,去创建server,这是一个需要学习的思路。
网友评论