GBP: contracts API fixed length of allowed ethertypes
[vpp.git] / extras / vom / vom / gbp_contract_cmds.cpp
1 /*
2  * Copyright (c) 2017 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 "vom/gbp_contract_cmds.hpp"
17 #include "vom/api_types.hpp"
18
19 namespace VOM {
20 namespace gbp_contract_cmds {
21
22 create_cmd::create_cmd(HW::item<bool>& item,
23                        sclass_t sclass,
24                        sclass_t dclass,
25                        const handle_t& acl,
26                        const gbp_contract::gbp_rules_t& gbp_rules,
27                        const gbp_contract::ethertype_set_t& allowed_ethertypes)
28   : rpc_cmd(item)
29   , m_sclass(sclass)
30   , m_dclass(dclass)
31   , m_acl(acl)
32   , m_gbp_rules(gbp_rules)
33   , m_allowed_ethertypes(allowed_ethertypes)
34 {
35 }
36
37 bool
38 create_cmd::operator==(const create_cmd& other) const
39 {
40   return ((m_acl == other.m_acl) && (m_sclass == other.m_sclass) &&
41           (m_dclass == other.m_dclass) && (m_gbp_rules == other.m_gbp_rules) &&
42           (m_allowed_ethertypes == other.m_allowed_ethertypes));
43 }
44
45 #define ARRAY_LEN(x) (sizeof(x) / sizeof(x[0]))
46
47 rc_t
48 create_cmd::issue(connection& con)
49 {
50   size_t n_rules = m_gbp_rules.size();
51   uint32_t ii = 0;
52
53   msg_t req(con.ctx(), n_rules, std::ref(*this));
54
55   auto& payload = req.get_request().get_payload();
56   payload.is_add = 1;
57   payload.contract.acl_index = m_acl.value();
58   payload.contract.sclass = m_sclass;
59   payload.contract.dclass = m_dclass;
60   payload.contract.n_rules = n_rules;
61   payload.contract.n_ether_types = m_allowed_ethertypes.size();
62
63   for (auto tt : m_allowed_ethertypes) {
64     payload.contract.allowed_ethertypes[ii] = tt.value();
65     ii++;
66     if (ii == ARRAY_LEN(payload.contract.allowed_ethertypes))
67       break;
68   }
69
70   ii = 0;
71   for (auto rule : m_gbp_rules) {
72     if (rule.action() == gbp_rule::action_t::REDIRECT)
73       payload.contract.rules[ii].action = GBP_API_RULE_REDIRECT;
74     else if (rule.action() == gbp_rule::action_t::PERMIT)
75       payload.contract.rules[ii].action = GBP_API_RULE_PERMIT;
76     else
77       payload.contract.rules[ii].action = GBP_API_RULE_DENY;
78
79     if (rule.nhs().hash_mode() == gbp_rule::hash_mode_t::SYMMETRIC)
80       payload.contract.rules[ii].nh_set.hash_mode = GBP_API_HASH_MODE_SYMMETRIC;
81     else if (rule.nhs().hash_mode() == gbp_rule::hash_mode_t::SRC_IP)
82       payload.contract.rules[ii].nh_set.hash_mode = GBP_API_HASH_MODE_SRC_IP;
83     else
84       payload.contract.rules[ii].nh_set.hash_mode = GBP_API_HASH_MODE_DST_IP;
85
86     const gbp_rule::next_hops_t& next_hops = rule.nhs().next_hops();
87     uint8_t jj = 0, nh_size = (next_hops.size() > 8) ? 8 : next_hops.size();
88
89     payload.contract.rules[ii].nh_set.n_nhs = nh_size;
90     for (auto nh : next_hops) {
91       to_api(nh.getIp(), payload.contract.rules[ii].nh_set.nhs[jj].ip);
92       to_api(nh.getMac(), payload.contract.rules[ii].nh_set.nhs[jj].mac);
93       payload.contract.rules[ii].nh_set.nhs[jj].bd_id = nh.getBdId();
94       payload.contract.rules[ii].nh_set.nhs[jj].rd_id = nh.getRdId();
95       jj++;
96     }
97     ++ii;
98   }
99
100   VAPI_CALL(req.execute());
101
102   return (wait());
103 }
104
105 std::string
106 create_cmd::to_string() const
107 {
108   std::ostringstream s;
109   s << "gbp-contract-create: " << m_hw_item.to_string()
110     << " sclass:" << m_sclass << " dclass:" << m_dclass << " acl:" << m_acl;
111   s << "[ethertype:";
112   for (auto e : m_allowed_ethertypes)
113     s << " " << e;
114   s << "]";
115
116   return (s.str());
117 }
118
119 delete_cmd::delete_cmd(HW::item<bool>& item, sclass_t sclass, sclass_t dclass)
120   : rpc_cmd(item)
121   , m_sclass(sclass)
122   , m_dclass(dclass)
123 {
124 }
125
126 bool
127 delete_cmd::operator==(const delete_cmd& other) const
128 {
129   return ((m_sclass == other.m_sclass) && (m_dclass == other.m_dclass));
130 }
131
132 rc_t
133 delete_cmd::issue(connection& con)
134 {
135   msg_t req(con.ctx(), 0, std::ref(*this));
136
137   auto& payload = req.get_request().get_payload();
138   payload.is_add = 0;
139   payload.contract.acl_index = ~0;
140   payload.contract.sclass = m_sclass;
141   payload.contract.dclass = m_dclass;
142
143   VAPI_CALL(req.execute());
144
145   return (wait());
146 }
147
148 std::string
149 delete_cmd::to_string() const
150 {
151   std::ostringstream s;
152   s << "gbp-contract-delete: " << m_hw_item.to_string()
153     << " sclass:" << m_sclass << " dclass:" << m_dclass;
154
155   return (s.str());
156 }
157
158 bool
159 dump_cmd::operator==(const dump_cmd& other) const
160 {
161   return (true);
162 }
163
164 rc_t
165 dump_cmd::issue(connection& con)
166 {
167   m_dump.reset(new msg_t(con.ctx(), std::ref(*this)));
168
169   VAPI_CALL(m_dump->execute());
170
171   wait();
172
173   return rc_t::OK;
174 }
175
176 std::string
177 dump_cmd::to_string() const
178 {
179   return ("gbp-contract-dump");
180 }
181
182 }; // namespace gbp_contract_cmds
183 }; // namespace VOM
184
185 /*
186  * fd.io coding-style-patch-verification: ON
187  *
188  * Local Variables:
189  * eval: (c-set-style "mozilla")
190  * End:
191  */