既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新
OAI高层结构
OAI L3主要分为RRC,NGAP,SCTP三部分:
- RRC负责无线资源管理,负责让gNB与UE建立连接,以及信令编解码。
- NGAP负责gNB与AMF建立连接,进行信令NGAP编解码并转发。
- SCTP类似于TCP,提供可靠的网络传输,负责AMF与gNB之间NGAP消息发送和接收。
在各模块之间,OAI使用了ITTI公共管理模块来负责规范实体管理,线程管理,队列管理,内存管理等,保证了各模块在资源使用上的规范性。
高层模块传输接口
OAI主程序分别调用itti_create_task()创建RRC、NGAP、SCTP以及PDCP主线程,线程创建时可进行CPU内核绑定(当前OAI程序未启用核绑定)。线程创建后,在各模块内使用while(1)循环调用 itti_receive_msg()从本模块消息队列中取数据,解码后按流程处理,发送消息时调用iiti_send_msg_to_task()将数据压入目标模块的队列。SCTP与AMF之间以SCTP连接进行消息收发。
ITTI实体线程创建
实体创建
实体创建实际就是OAI各模块task对象的创建,以及相应的线程创建,每个模块一般为一个主线程,以及包含可能存在的辅助线程。线程内while()循环保证了实体始终处于运行状态,这一点属于常规方法。
int itti\_create\_task(task\_id\_t task_id,
void \*(\*start_routine)(void \*),
void \*args_p) {
task\_list\_t \*t=tasks[task_id];
threadCreate (&t->thread, start_routine, args_p, (char \*)itti\_get\_task\_name(task_id),-1,OAI_PRIORITY_RT);
LOG\_I(TMR,"Created Posix thread %s\n", itti\_get\_task\_name(task_id) );
return 0;
}
main调用itti_create_task函数来创建实体,其中
task_id:模块实体ID,如TASK_RRC_GNB
start_routine:创建的线程函数名
args_p:传递的参数,OAI里均为NULL
代码中的tasks数组为各个实体的集合,包含了每个实体公共属性,方便随时取用。
threadCreate函数负责线程创建
线程创建
线程创建流程如下,创建步骤见注释
void threadCreate(pthread\_t\* t, void \* (\*func)(void\*), void \* param, char\* name, int affinity, int priority){
pthread\_attr\_t attr;
int ret;
int settingPriority = 1;
//初始化线程对象的属性
ret=pthread\_attr\_init(&attr);
AssertFatal(ret==0,"ret: %d, errno: %d\n",ret, errno);
//设置线程为分离状态
ret=pthread\_attr\_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
AssertFatal(ret==0,"ret: %d, errno: %d\n",ret, errno);
//设置线程不继承父线程的调度策略
ret=pthread\_attr\_setinheritsched(&attr, PTHREAD_EXPLICIT_SCHED);
AssertFatal(ret==0,"ret: %d, errno: %d\n",ret, errno);
if (checkIfFedoraDistribution())
if (checkIfGenericKernelOnFedora())
if (checkIfInsideContainer())
settingPriority = 0;
/\*SCHED\_OAI,其宏定义为#define SCHED\_OAI SCHED\_RR,即线程采用轮询式调度
各模块可以根据需要确定自己的优先级\*/
if (settingPriority) {
ret=pthread\_attr\_setschedpolicy(&attr, SCHED_OAI);
AssertFatal(ret==0,"ret: %d, errno: %d\n",ret, errno);
if(priority<sched\_get\_priority\_min(SCHED_OAI) || priority>sched\_get\_priority\_max(SCHED_FIFO)) {
LOG\_E(TMR,"Prio not possible: %d, min is %d, max: %d, forced in the range\n",
priority,
sched\_get\_priority\_min(SCHED_OAI),
sched\_get\_priority\_max(SCHED_OAI));
if(priority<sched\_get\_priority\_min(SCHED_OAI))
priority=sched\_get\_priority\_min(SCHED_OAI);
if(priority>sched\_get\_priority\_max(SCHED_OAI))
priority=sched\_get\_priority\_max(SCHED_OAI);
}
AssertFatal(priority<=sched\_get\_priority\_max(SCHED_OAI),"");
struct sched\_param sparam={0};
sparam.sched_priority = priority;
//设置线程优先级
ret=pthread\_attr\_setschedparam(&attr, &sparam);
AssertFatal(ret==0,"ret: %d, errno: %d\n",ret, errno);
}
//利用前面设置的属性变量创建线程
ret=pthread\_create(t, &attr, func, param);
AssertFatal(ret==0,"ret: %d, errno: %d\n",ret, errno);
//设置线程名
pthread\_setname\_np(\*t, name);
if (affinity != -1 ) {
cpu\_set\_t cpuset;
CPU\_ZERO(&cpuset);
CPU\_SET(affinity, &cpuset);
AssertFatal( pthread\_setaffinity\_np(\*t, sizeof(cpu\_set\_t), &cpuset) == 0, "Error setting processor affinity");
}
//销毁线程属性变量
![img](https://img-blog.csdnimg.cn/img_convert/d7213d10b157ff7d6a6976c95ae2313d.png)
![img](https://img-blog.csdnimg.cn/img_convert/941cfc3df124f4b50ae9a03c3fb9543d.png)
![img](https://img-blog.csdnimg.cn/img_convert/779fce871ea179ebef448aa00e63d1bb.png)
**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**