blob: 59e2973b5893ed747aa4c3b59892938c9a7a9452 [file] [log] [blame]
Neale Rannsd792d9c2017-10-21 10:53:20 -07001/*
2 * Copyright (c) 2016 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16#include <vnet/bier/bier_fmask_db.h>
17#include <vnet/bier/bier_fmask.h>
18
19/**
20 * Global Table of fmask objects
21 * The key into this table includes the table's key and the fmask's key,
22 * so there could be a DB per-table. But it is more efficient
23 * at forwarding time to extract the fmask from a single global table
24 * which is hot in dcache.
25 *
26 * The table's key is part of this DB key, since the fmasks therein build up
27 * their forwarding mask based on the routes that resolve through
28 * it, so cross polination would be bad.
29 */
30typedef struct bier_fmask_db_t_ {
31 /**
32 * hash table for underlying storage
33 */
Neale Ranns586479a2018-06-07 02:08:07 -070034 uword *bfdb_hash;
Neale Rannsd792d9c2017-10-21 10:53:20 -070035
36 /**
37 * Pool for memory
38 */
39 struct bier_fmask_t_ *bfdb_pool;
40} bier_fmask_db_t;
41
42/**
Neale Rannsd792d9c2017-10-21 10:53:20 -070043 * Single fmask DB
44 */
45static bier_fmask_db_t bier_fmask_db;
46
47
48u32
49bier_fmask_get_index (const bier_fmask_t *bfm)
50{
51 return (bfm - bier_fmask_db.bfdb_pool);
52}
53
Neale Ranns91286372017-12-05 13:24:04 -080054static void
55bier_fmask_db_mk_key (index_t bti,
56 const fib_route_path_t *rpath,
57 bier_fmask_id_t *key)
58{
59 /*
60 * Depending on what the ID is there may be padding.
61 * This key will be memcmp'd in the mhash, so make sure it's all 0
62 */
63 memset(key, 0, sizeof(*key));
64
65 /*
66 * Pick the attributes from the path that make the FMask unique
67 */
68 if (FIB_ROUTE_PATH_UDP_ENCAP & rpath->frp_flags)
69 {
70 key->bfmi_id = rpath->frp_udp_encap_id;
Neale Ranns586479a2018-06-07 02:08:07 -070071 key->bfmi_nh_type = BIER_NH_UDP;
Neale Ranns91286372017-12-05 13:24:04 -080072 }
73 else
74 {
Neale Ranns91286372017-12-05 13:24:04 -080075 memcpy(&key->bfmi_nh, &rpath->frp_addr, sizeof(rpath->frp_addr));
Neale Ranns586479a2018-06-07 02:08:07 -070076 key->bfmi_nh_type = BIER_NH_IP;
Neale Ranns91286372017-12-05 13:24:04 -080077 }
78 if (NULL == rpath->frp_label_stack)
79 {
80 key->bfmi_hdr_type = BIER_HDR_O_OTHER;
81 }
82 else
83 {
84 key->bfmi_hdr_type = BIER_HDR_O_MPLS;
85 }
Neale Ranns586479a2018-06-07 02:08:07 -070086 key->bfmi_bti = bti;
Neale Ranns91286372017-12-05 13:24:04 -080087}
88
89u32
90bier_fmask_db_find (index_t bti,
91 const fib_route_path_t *rpath)
92{
93 bier_fmask_id_t fmid;
94 uword *p;
95
96 bier_fmask_db_mk_key(bti, rpath, &fmid);
Neale Ranns586479a2018-06-07 02:08:07 -070097 p = hash_get_mem(bier_fmask_db.bfdb_hash, &fmid);
Neale Ranns91286372017-12-05 13:24:04 -080098
99 if (NULL != p)
100 {
101 return (p[0]);
102 }
103
104 return (INDEX_INVALID);
105}
106
Neale Rannsd792d9c2017-10-21 10:53:20 -0700107u32
108bier_fmask_db_find_or_create_and_lock (index_t bti,
Neale Rannsd792d9c2017-10-21 10:53:20 -0700109 const fib_route_path_t *rpath)
110{
Neale Ranns91286372017-12-05 13:24:04 -0800111 bier_fmask_id_t fmid;
Neale Rannsd792d9c2017-10-21 10:53:20 -0700112 u32 index;
113 uword *p;
114
Neale Ranns91286372017-12-05 13:24:04 -0800115 bier_fmask_db_mk_key(bti, rpath, &fmid);
Neale Ranns586479a2018-06-07 02:08:07 -0700116 p = hash_get_mem(bier_fmask_db.bfdb_hash, &fmid);
Neale Rannsd792d9c2017-10-21 10:53:20 -0700117
118 if (NULL == p)
119 {
Neale Ranns91286372017-12-05 13:24:04 -0800120 bier_fmask_t *bfm;
Neale Rannsd792d9c2017-10-21 10:53:20 -0700121 /*
122 * adding a new fmask object
123 */
Neale Ranns91286372017-12-05 13:24:04 -0800124 index = bier_fmask_create_and_lock(&fmid, rpath);
125 bfm = bier_fmask_get(index);
Neale Ranns586479a2018-06-07 02:08:07 -0700126 hash_set_mem(bier_fmask_db.bfdb_hash, bfm->bfm_id, index);
Neale Rannsd792d9c2017-10-21 10:53:20 -0700127 }
128 else
129 {
130 index = p[0];
131 bier_fmask_lock(index);
132 }
133
134 return (index);
135}
136
Neale Rannsd792d9c2017-10-21 10:53:20 -0700137void
Neale Ranns91286372017-12-05 13:24:04 -0800138bier_fmask_db_remove (const bier_fmask_id_t *fmid)
Neale Rannsd792d9c2017-10-21 10:53:20 -0700139{
Neale Rannsd792d9c2017-10-21 10:53:20 -0700140 uword *p;
141
Neale Ranns586479a2018-06-07 02:08:07 -0700142 p = hash_get_mem(bier_fmask_db.bfdb_hash, fmid);
Neale Rannsd792d9c2017-10-21 10:53:20 -0700143
144 if (NULL == p) {
145 /*
146 * remove a non-exitant entry - oops
147 */
148 ASSERT (!"remove non-existant fmask");
149 } else {
Neale Ranns586479a2018-06-07 02:08:07 -0700150 hash_unset(bier_fmask_db.bfdb_hash, fmid);
Neale Rannsd792d9c2017-10-21 10:53:20 -0700151 }
152}
153
Neale Ranns586479a2018-06-07 02:08:07 -0700154void
155bier_fmask_db_walk (bier_fmask_walk_fn_t fn, void *ctx)
156{
157 CLIB_UNUSED (bier_fmask_id_t *fmid);
158 uword *bfmi;
159
160 hash_foreach(fmid, bfmi, bier_fmask_db.bfdb_hash,
161 ({
162 if (WALK_STOP == fn(*bfmi, ctx))
163 break;
164 }));
165}
166
Neale Rannsd792d9c2017-10-21 10:53:20 -0700167clib_error_t *
168bier_fmask_db_module_init (vlib_main_t *vm)
169{
Neale Ranns586479a2018-06-07 02:08:07 -0700170 bier_fmask_db.bfdb_hash = hash_create_mem(0,
171 sizeof(bier_fmask_id_t),
172 sizeof(index_t));
Neale Rannsd792d9c2017-10-21 10:53:20 -0700173
174 return (NULL);
175}
176
177VLIB_INIT_FUNCTION (bier_fmask_db_module_init);