bestboyxie 励志做一名能帮助到他人的程序员,如果你觉得这篇文章对你有帮助,麻烦你
点赞
DPDK的ring是一个线程安全的队列 支持单生产者单消费模型 同时也支持多生产者多消费者模型
一、单生产者单消费者模型 二、多生产者对消费者模型
多生产者多消费者应该至少是由原子变量或者锁来完成的,而单生产者单消费者则可以做到完全无锁,效率感觉单生产者单消费应该是最高的;不然这个单生产者单消费者就没有存在的必要了;、 效率:单生产者单消费者最高 本文章的内容是dpdk的example代码梳理出来的;
DPDK的ring特点 1。dpdk来保证高效; 2。多线程多进程都可以用; 3。常用来跨进程、线程间通信用; 所谓队列其实就是fifo。先进先出; dpdk的ring入队列的是
一个指针,配套使用dpdk的共享内存,很容易实现进程间低消耗的通讯;
一般流程 创建ring——》入队列-》出队列 一、创建ring
struct rte_ring * rte_ring_create(const char *name, unsigned count, int socket_id,unsigned flags)
第一个参数:ring的名字,dpdk 的所有内存结构都可以通过名字来标记; 第二个参数:ring的个数 count 貌似这个count记得要设置为2的N次方。不然会失败; 第三个参数:socketid核的id。一般如果创建自己用,使用rte_socket_id()函数就OK了。如果是跨核访问使用对应socketid就好了。(这个参数在多CPU平台上有用); 第四个参数:flag。主要指单生产者 (RING_F_SP_ENQ )单消费(RING_F_SC_DEQ),如果是0就是多生产者多消费;
例子
struct rte_ring * ring_p = rte_ring_create("myring",1024,0,RING_F_SP_ENQ |RING_F_SC_DEQ)
二、查找ring 如果ring是两个进程共享的,这个时候非创建者只需要使用查找ring的函数查找到ring就可以使用了;
struct rte_ring * rte_ring_lookup(const char *name)
例子:
struct rte_ring * ring_p = rte_ring_lookup("myring")
三、入/出队列 入/出队列有两种。一种是单个单个的入,一种是一次性入一群;
int rte_ring_enqueue(struct rte_ring *ring,void *data)
<<------>>
int rte_ring_dequeue(struct rte_ring *ring,void **data)
int rte_ring_enqueue_bulk(struct rte_ring *ring,void **data,int count)
<<------>>
int rte_ring_dequeue_bulk(struct rte_ring *ring,void **data,int count)
示例使用 rte_ring_enqueue /rte_ring_dequeue 返回值为0成功,否则失败; 进程1 struct rte_mbuf *mbuf=xxxx; rte_ring_enqueue(ringp,mbuf); 进程2 struct rte_mbuf *mbuf=NULL; rte_ring_dequeue(ringp,&mbuf);
rte_ring_enqueue_bulk 返回值为实际入队列的个数 rte_ring_dequeue_bulk返回值为0,则全部成功。否则一个都不会成功 进程1
struct rte_mbuf *mbuf[100];
.....
int count = rte_ring_enqueue_bulk(ringp,(void **)mbuf,100);
if(count != 100){
for(;
count<100;
count++){
rte_mbuf_free(mbuf[count]);
}
}
进程2
int count= rte_ring_count(ringp);
struct rte_mbuf *mbuf[100];
if(rte_ring_dequeue_bulk(ringp,(void **)mbuf,100) == 0){
//do something
}
四、删除ring; rte_ring_free(ringp); 五、ring状态获取 查看已经使用的count数目 rte_ring_count() 查看空闲的count数据 rte_ring_free_count() 一键判断空/满 rte_ring_empty/rte_ring_full
所有的API都在 rte_ring.h中,上下文都比较详细 参考代码可以查看 examples\multi_process 下面的代码即可
高级用法。 dpdk ring 设置水位线。用于队尾丢弃策略的控制;
如果觉得好, 请下面顶