f-stack/adapter/ff_so_zone.c

171 lines
4.3 KiB
C
Raw Normal View History

2023-04-06 07:02:39 +00:00
#include <stdio.h>
#include <string.h>
#include <rte_eal.h>
#include <rte_memzone.h>
#include "ff_config.h"
#include "ff_socket_ops.h"
#define SOCKET_OPS_ZONE_NAME "ff_socket_ops_zone_%d"
#define SOCKET_OPS_CONTEXT_MAX_NUM 32
#define SOCKET_OPS_CONTEXT_NAME_SIZE 32
#define SOCKET_OPS_CONTEXT_NAME "ff_so_context_"
static uint16_t ff_max_so_context = SOCKET_OPS_CONTEXT_MAX_NUM;
struct ff_socket_ops_zone *ff_so_zone;
int
ff_set_max_so_context(uint16_t count)
{
if (ff_so_zone) {
ERR_LOG("Can not set: memzone has inited\n");
return -1;
}
/*if (rte_eal_process_type() != RTE_PROC_PRIMARY) {
ERR_LOG("Can not set: process is not primary\n");
return 1;
}*/
ff_max_so_context = count;
return 0;
}
int
ff_create_so_memzone()
{
if (ff_so_zone) {
ERR_LOG("Can not create memzone: memzone has inited\n");
return -1;
}
if (rte_eal_process_type() == RTE_PROC_PRIMARY) {
uint16_t i, proc_id;
for (proc_id = 0; proc_id < ff_global_cfg.dpdk.nb_procs; proc_id++) {
struct ff_socket_ops_zone *so_zone_tmp;
const struct rte_memzone *mz;
char zn[64];
size_t zone_size = sizeof(struct ff_socket_ops_zone) +
sizeof(struct ff_so_context) * ff_max_so_context;
snprintf(zn, sizeof(zn), SOCKET_OPS_ZONE_NAME, proc_id);
ERR_LOG("To create memzone:%s\n", zn);
mz = rte_memzone_reserve(zn, zone_size, rte_socket_id(), 0);
if (mz == NULL) {
ERR_LOG("Cannot reserve memory zone:%s\n", zn);
return -1;
}
memset(mz->addr, 0, zone_size);
so_zone_tmp = mz->addr;
rte_spinlock_init(&so_zone_tmp->lock);
so_zone_tmp->count = ff_max_so_context;
so_zone_tmp->free = so_zone_tmp->count;
so_zone_tmp->sc = (struct ff_so_context *)(so_zone_tmp + 1);
for (i = 0; i < ff_max_so_context; i++) {
struct ff_so_context *sc = &so_zone_tmp->sc[i];
rte_spinlock_init(&sc->lock);
sc->status = FF_SC_IDLE;
sc->inuse = 0;
if (sem_init(&sc->wait_sem, 1, 0) == -1) {
ERR_LOG("Initialize semaphore failed:%d\n", errno);
return -1;
}
}
if (proc_id == 0) {
ff_so_zone = so_zone_tmp;
}
}
}else {
const struct rte_memzone *mz;
char zn[64];
snprintf(zn, sizeof(zn), SOCKET_OPS_ZONE_NAME, ff_global_cfg.dpdk.proc_id);
ERR_LOG("To lookup memzone:%s\n", zn);
mz = rte_memzone_lookup(zn);
if (mz == NULL) {
ERR_LOG("Lookup memory zone:%s failed\n", zn);
return -1;
}
ff_so_zone = mz->addr;
}
return 0;
}
struct ff_so_context *
ff_attach_so_context(int proc_id)
{
struct ff_so_context *sc = NULL;
uint16_t i;
if (ff_so_zone == NULL) {
const struct rte_memzone *mz;
char zn[64];
snprintf(zn, sizeof(zn), SOCKET_OPS_ZONE_NAME, proc_id);
ERR_LOG("To lookup memzone:%s\n", zn);
mz = rte_memzone_lookup(zn);
if (mz == NULL) {
ERR_LOG("Lookup memory zone:%s failed\n", zn);
return NULL;
}
ff_so_zone = mz->addr;
}
rte_spinlock_lock(&ff_so_zone->lock);
if (ff_so_zone->free == 0) {
ERR_LOG("Attach memzone failed: no free context\n");
rte_spinlock_unlock(&ff_so_zone->lock);
return NULL;
}
/* i从count-free开始尝试直到一轮结束 */
for (i = 0; i < ff_so_zone->count; i++) {
sc = &ff_so_zone->sc[i];
if (sc->inuse == 0) {
sc->inuse = 1;
rte_spinlock_init(&sc->lock);
sc->status = FF_SC_IDLE;
ff_so_zone->free--;
break;
}
}
DEBUG_LOG("attach sc:%p, i:%d\n", sc, i);
rte_spinlock_unlock(&ff_so_zone->lock);
return sc;
}
void
ff_detach_so_context(struct ff_so_context *sc)
{
if (ff_so_zone == NULL || sc == NULL) {
return;
}
rte_spinlock_lock(&ff_so_zone->lock);
sc->inuse = 0;
ff_so_zone->free++;
DEBUG_LOG("detach sc:%p, status:%d, ops:%d\n", sc, sc->status, sc->ops);
rte_spinlock_unlock(&ff_so_zone->lock);
}