"src/include/migraph/verify.hpp" did not exist on "3f66a1edac3f789a741f1f31f9c355caafb5947f"
i40e_adminq.cc 14.2 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#include <stdlib.h>
#include <string.h>
#include <cassert>
#include <iostream>

#include "i40e_bm.h"

#include "i40e_base_wrapper.h"

using namespace i40e;

extern nicbm::Runner *runner;

queue_admin_tx::queue_admin_tx(i40e_bm &dev_, uint64_t &reg_base_,
        uint32_t &reg_len_, uint32_t &reg_head_, uint32_t &reg_tail_)
16
    : queue_base("atx", reg_head_, reg_tail_), dev(dev_), reg_base(reg_base_),
17
18
19
    reg_len(reg_len_)
{
    desc_len = 32;
20
    ctxs_init();
21
22
}

23
24
25
26
27
28
29
30
31
32
33
queue_base::desc_ctx &queue_admin_tx::desc_ctx_create()
{
    return *new admin_desc_ctx(*this, dev);
}

void queue_admin_tx::reg_updated()
{
    base = reg_base;
    len = (reg_len & I40E_GL_ATQLEN_ATQLEN_MASK) >> I40E_GL_ATQLEN_ATQLEN_SHIFT;

    if (!enabled  && (reg_len & I40E_GL_ATQLEN_ATQENABLE_MASK)) {
34
35
36
37
#ifdef DEBUG_ADMINQ
        std::cerr << "atq: enable base=" << base << " len=" << len <<
            std::endl;
#endif
38
39
        enabled = true;
    } else if (enabled && !(reg_len & I40E_GL_ATQLEN_ATQENABLE_MASK)) {
40
41
42
#ifdef DEBUG_ADMINQ
        std::cerr << "atq: disable" << std::endl;
#endif
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
        enabled = false;
    }

    queue_base::reg_updated();
}

queue_admin_tx::admin_desc_ctx::admin_desc_ctx(queue_admin_tx &queue_,
        i40e_bm &dev_)
    : i40e::queue_base::desc_ctx(queue_), aq(queue_), dev(dev_)
{
    d = reinterpret_cast <struct i40e_aq_desc *> (desc);
}

void queue_admin_tx::admin_desc_ctx::data_written(uint64_t addr, size_t len)
{
    processed();
}

void queue_admin_tx::admin_desc_ctx::desc_compl_prepare(uint16_t retval,
62
63
64
65
66
67
68
        uint16_t extra_flags)
{
    d->flags &= ~0x1ff;
    d->flags |= I40E_AQ_FLAG_DD | I40E_AQ_FLAG_CMP | extra_flags;
    if (retval)
        d->flags |= I40E_AQ_FLAG_ERR;
    d->retval = retval;
Antoine Kaufmann's avatar
Antoine Kaufmann committed
69
70
71
72
73

#ifdef DEBUG_ADMINQ
    std::cerr << "atq: desc_compl_prepare index=" << index << " retval=" <<
        retval << std::endl;
#endif
74
75
}

76
77
void queue_admin_tx::admin_desc_ctx::desc_complete(uint16_t retval,
        uint16_t extra_flags)
78
{
79
80
    desc_compl_prepare(retval, extra_flags);
    processed();
81
82
}

83
84
void queue_admin_tx::admin_desc_ctx::desc_complete_indir(uint16_t retval,
        const void *data, size_t len, uint16_t extra_flags, bool ignore_datalen)
85
{
86
    if (!ignore_datalen && len > d->datalen) {
87
88
89
90
91
92
        std::cerr << "queue_admin_tx::desc_complete_indir: data too long ("
            << len << ") got buffer for (" << d->datalen << ")" << std::endl;
        abort();
    }
    d->datalen = len;

93
    desc_compl_prepare(retval, extra_flags);
94
95
96

    uint64_t addr = d->params.external.addr_low |
        (((uint64_t) d->params.external.addr_high) << 32);
97
    data_write(addr, len, data);
98
99
}

100
void queue_admin_tx::admin_desc_ctx::prepare()
101
{
102
103
104
    if ((d->flags & I40E_AQ_FLAG_RD)) {
        uint64_t addr = d->params.external.addr_low |
            (((uint64_t) d->params.external.addr_high) << 32);
105
106
107
108
#ifdef DEBUG_ADMINQ
        std::cerr << "atq: desc with buffer opc=" << d->opcode << " addr=" <<
            addr << std::endl;
#endif
109
110
111
112
113
        data_fetch(addr, d->datalen);
    } else {
        prepared();
    }
}
114

115
116
void queue_admin_tx::admin_desc_ctx::process()
{
117
118
119
#ifdef DEBUG_ADMINQ
    std::cerr << "atq: descriptor " << index << " fetched" << std::endl;
#endif
120
121

    if (d->opcode == i40e_aqc_opc_get_version) {
122
123
124
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  get version" << std::endl;
#endif
125
126
127
128
129
130
131
132
133
        struct i40e_aqc_get_version *gv =
            reinterpret_cast<struct i40e_aqc_get_version *>(d->params.raw);
        gv->rom_ver = 0;
        gv->fw_build = 0;
        gv->fw_major = 0;
        gv->fw_minor = 0;
        gv->api_major = I40E_FW_API_VERSION_MAJOR;
        gv->api_minor = I40E_FW_API_VERSION_MINOR_X710;

134
        desc_complete(0);
135
    } else if (d->opcode == i40e_aqc_opc_request_resource) {
136
137
138
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  request resource" << std::endl;
#endif
139
140
141
142
        struct i40e_aqc_request_resource *rr =
            reinterpret_cast<struct i40e_aqc_request_resource *>(
                    d->params.raw);
        rr->timeout = 180000;
143
144
145
146
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:    res_id=" << rr->resource_id << std::endl;
        std::cerr << "atq:    res_nu=" << rr->resource_number << std::endl;
#endif
147
        desc_complete(0);
148
    } else if (d->opcode == i40e_aqc_opc_release_resource) {
149
150
151
152
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  release resource" << std::endl;
#endif
#ifdef DEBUG_ADMINQ
153
154
155
        struct i40e_aqc_request_resource *rr =
            reinterpret_cast<struct i40e_aqc_request_resource *>(
                    d->params.raw);
156
157
158
        std::cerr << "atq:    res_id=" << rr->resource_id << std::endl;
        std::cerr << "atq:    res_nu=" << rr->resource_number << std::endl;
#endif
159
        desc_complete(0);
160
    } else if (d->opcode == i40e_aqc_opc_clear_pxe_mode)  {
161
162
163
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  clear PXE mode" << std::endl;
#endif
164
        dev.regs.gllan_rctl_0 &= ~I40E_GLLAN_RCTL_0_PXE_MODE_MASK;
165
        desc_complete(0);
166
167
168
    } else if (d->opcode == i40e_aqc_opc_list_func_capabilities ||
            d->opcode == i40e_aqc_opc_list_dev_capabilities)
    {
169
170
171
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  get dev/fun caps" << std::endl;
#endif
172
173
174
175
176
177
178
179
180
        struct i40e_aqc_list_capabilites *lc =
            reinterpret_cast<struct i40e_aqc_list_capabilites *>(
                    d->params.raw);

        struct i40e_aqc_list_capabilities_element_resp caps[] = {
            { I40E_AQ_CAP_ID_RSS, 1, 0, 512, 6, 0, {} },
            { I40E_AQ_CAP_ID_RXQ, 1, 0, dev.NUM_QUEUES, 0, 0, {} },
            { I40E_AQ_CAP_ID_TXQ, 1, 0, dev.NUM_QUEUES, 0, 0, {} },
            { I40E_AQ_CAP_ID_MSIX, 1, 0, dev.NUM_PFINTS, 0, 0, {} },
181
182
            { I40E_AQ_CAP_ID_VSI, 1, 0, dev.NUM_VSIS, 0, 0, {} },
            { I40E_AQ_CAP_ID_DCB, 1, 0, 1, 1, 1, {} },
183
184
185
186
        };
        size_t num_caps = sizeof(caps) / sizeof(caps[0]);

        if (sizeof(caps) <= d->datalen) {
187
188
189
#ifdef DEBUG_ADMINQ
            std::cerr << "atq:    data fits" << std::endl;
#endif
190
191
            // data fits within the buffer
            lc->count = num_caps;
192
            desc_complete_indir(0, caps, sizeof(caps));
193
        } else {
194
195
196
#ifdef DEBUG_ADMINQ
            std::cerr << "atq:    data doesn't fit" << std::endl;
#endif
197
198
            // data does not fit
            d->datalen = sizeof(caps);
199
            desc_complete(I40E_AQ_RC_ENOMEM);
200
201
        }
    } else if (d->opcode == i40e_aqc_opc_lldp_stop) {
202
203
204
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  lldp stop" << std::endl;
#endif
205
        desc_complete(0);
206
    } else if (d->opcode == i40e_aqc_opc_mac_address_read) {
207
208
209
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  read mac" << std::endl;
#endif
210
211
212
213
214
215
        struct i40e_aqc_mac_address_read *ar =
            reinterpret_cast<struct i40e_aqc_mac_address_read *>(
                    d->params.raw);

        struct i40e_aqc_mac_address_read_data ard;
        uint64_t mac = runner->get_mac_addr();
216
217
218
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:    mac = " << mac << std::endl;
#endif
219
220
221
222
        memcpy(ard.pf_lan_mac, &mac, 6);
        memcpy(ard.port_mac, &mac, 6);

        ar->command_flags = I40E_AQC_LAN_ADDR_VALID | I40E_AQC_PORT_ADDR_VALID;
223
        desc_complete_indir(0, &ard, sizeof(ard));
224
    } else if (d->opcode == i40e_aqc_opc_get_phy_abilities) {
225
226
227
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  get phy abilities" << std::endl;
#endif
228
229
230
231
232
233
234
235
236
237
238
239
        struct i40e_aq_get_phy_abilities_resp par;
        memset(&par, 0, sizeof(par));

        par.phy_type = (1ULL << I40E_PHY_TYPE_40GBASE_CR4_CU);
        par.link_speed = I40E_LINK_SPEED_40GB;
        par.abilities = I40E_AQ_PHY_LINK_ENABLED |
            I40E_AQ_PHY_AN_ENABLED;
        par.eee_capability = 0;

        d->params.external.param0 = 0;
        d->params.external.param1 = 0;

240
        desc_complete_indir(0, &par, sizeof(par), 0, true);
241
    } else if (d->opcode == i40e_aqc_opc_get_link_status) {
242
243
244
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  link status" << std::endl;
#endif
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
        struct i40e_aqc_get_link_status *gls =
            reinterpret_cast<struct i40e_aqc_get_link_status *>(
                    d->params.raw);

        gls->command_flags &= I40E_AQ_LSE_IS_ENABLED; // should actually return
                                                      // status of link status
                                                      // notification
        gls->phy_type = I40E_PHY_TYPE_40GBASE_CR4_CU;
        gls->link_speed = I40E_LINK_SPEED_40GB;
        gls->link_info = I40E_AQ_LINK_UP_FUNCTION | I40E_AQ_LINK_UP_PORT |
            I40E_AQ_MEDIA_AVAILABLE | I40E_AQ_SIGNAL_DETECT;
        gls->an_info = I40E_AQ_AN_COMPLETED | I40E_AQ_LP_AN_ABILITY; // might need qualified module
        gls->ext_info = 0;
        gls->loopback = I40E_AQ_LINK_POWER_CLASS_4 << I40E_AQ_PWR_CLASS_SHIFT_LB;
        gls->max_frame_size = dev.MAX_MTU;
        gls->config = I40E_AQ_CONFIG_CRC_ENA;

262
        desc_complete(0);
263
    } else if (d->opcode == i40e_aqc_opc_get_switch_config) {
264
265
266
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  get switch config" << std::endl;
#endif
267
268
269
        struct i40e_aqc_switch_seid *sw = reinterpret_cast<
            struct i40e_aqc_switch_seid *>(d->params.raw);
        struct i40e_aqc_get_switch_config_header_resp hr;
270
        /* Not sure why dpdk doesn't like this?
271
        struct i40e_aqc_switch_config_element_resp els[] = {
272
273
            // EMC
            { I40E_AQ_SW_ELEM_TYPE_EMP, I40E_AQ_SW_ELEM_REV_1, 1, 513, 0, {},
274
                I40E_AQ_CONN_TYPE_REGULAR, 0, 0},
275
276
            // MAC
            { I40E_AQ_SW_ELEM_TYPE_MAC, I40E_AQ_SW_ELEM_REV_1, 2, 0, 0, {},
277
278
                I40E_AQ_CONN_TYPE_REGULAR, 0, 0},
            // PF
279
280
281
282
283
284
285
286
287
288
289
290
            { I40E_AQ_SW_ELEM_TYPE_PF, I40E_AQ_SW_ELEM_REV_1, 16, 512, 0, {},
                I40E_AQ_CONN_TYPE_REGULAR, 0, 0},
            // VSI PF
            { I40E_AQ_SW_ELEM_TYPE_VSI, I40E_AQ_SW_ELEM_REV_1, 512, 2, 16, {},
                I40E_AQ_CONN_TYPE_REGULAR, 0, 0},
            // VSI PF
            { I40E_AQ_SW_ELEM_TYPE_VSI, I40E_AQ_SW_ELEM_REV_1, 513, 2, 1, {},
                I40E_AQ_CONN_TYPE_REGULAR, 0, 0},
        };*/
        struct i40e_aqc_switch_config_element_resp els[] = {
            // VSI PF
            { I40E_AQ_SW_ELEM_TYPE_VSI, I40E_AQ_SW_ELEM_REV_1, 512, 2, 16, {},
291
292
293
                I40E_AQ_CONN_TYPE_REGULAR, 0, 0},
        };

294

295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
        // find start idx
        size_t cnt = sizeof(els) / sizeof(els[0]);
        size_t first = 0;
        for (first = 0; first < cnt && els[first].seid < sw->seid; first++);

        // figure out how many fit in the buffer
        size_t max = (d->datalen - sizeof(hr)) / sizeof(els[0]);
        size_t report = cnt - first;
        if (report > max) {
            report = max;
            sw->seid = els[first + report].seid;
        } else {
            sw->seid = 0;
        }

        // prepare header
        memset(&hr, 0, sizeof(hr));
        hr.num_reported = report;
        hr.num_total = cnt;
314
315
316
317
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:    report=" << report << " cnt=" << cnt <<
            "  seid=" << sw->seid << std::endl;
#endif
318
319
320
321
322
323
324

        // create temporary contiguous buffer
        size_t buflen = sizeof(hr) + sizeof(els[0]) * report;
        uint8_t buf[buflen];
        memcpy(buf, &hr, sizeof(hr));
        memcpy(buf + sizeof(hr), els + first, sizeof(els[0]) * report);

325
        desc_complete_indir(0, buf, buflen);
326
    } else if (d->opcode == i40e_aqc_opc_set_switch_config) {
327
328
329
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  set switch config" << std::endl;
#endif
330
331
332
333
334
335
        /* TODO: lots of interesting things here like l2 filtering etc. that are
         * relevant.
        struct i40e_aqc_set_switch_config *sc =
            reinterpret_cast<struct i40e_aqc_set_switch_config *>(
                    d->params.raw);
        */
336
        desc_complete(0);
337
    } else if (d->opcode == i40e_aqc_opc_get_vsi_parameters) {
338
339
340
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  get vsi parameters" << std::endl;
#endif
Antoine Kaufmann's avatar
Antoine Kaufmann committed
341
        /*struct i40e_aqc_add_get_update_vsi *v =
342
            reinterpret_cast<struct i40e_aqc_add_get_update_vsi *>(
Antoine Kaufmann's avatar
Antoine Kaufmann committed
343
                    d->params.raw);*/
344
345
346
347
348
349
350

        struct i40e_aqc_vsi_properties_data pd;
        memset(&pd, 0, sizeof(pd));
        pd.valid_sections |= I40E_AQ_VSI_PROP_SWITCH_VALID |
            I40E_AQ_VSI_PROP_QUEUE_MAP_VALID |
            I40E_AQ_VSI_PROP_QUEUE_OPT_VALID |
            I40E_AQ_VSI_PROP_SCHED_VALID;
351
        desc_complete_indir(0, &pd, sizeof(pd));
352
    } else if (d->opcode == i40e_aqc_opc_update_vsi_parameters) {
353
354
355
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  update vsi parameters" << std::endl;
#endif
356
        /* TODO */
357
        desc_complete(0);
358
    } else if (d->opcode == i40e_aqc_opc_set_dcb_parameters) {
359
360
361
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  set dcb parameters" << std::endl;
#endif
362
        /* TODO */
363
        desc_complete(0);
364
    } else if (d->opcode == i40e_aqc_opc_configure_vsi_bw_limit) {
365
366
367
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  configure vsi bw limit" << std::endl;
#endif
368
        desc_complete(0);
369
    } else if (d->opcode == i40e_aqc_opc_query_vsi_bw_config) {
370
371
372
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  query vsi bw config" << std::endl;
#endif
373
374
375
376
        struct i40e_aqc_query_vsi_bw_config_resp bwc;
        memset(&bwc, 0, sizeof(bwc));
        for (size_t i = 0; i < 8; i++)
            bwc.qs_handles[i] = 0xffff;
377
        desc_complete_indir(0, &bwc, sizeof(bwc));
378
    } else if (d->opcode == i40e_aqc_opc_query_vsi_ets_sla_config) {
379
380
381
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  query vsi ets sla config" << std::endl;
#endif
382
383
384
385
        struct i40e_aqc_query_vsi_ets_sla_config_resp sla;
        memset(&sla, 0, sizeof(sla));
        for (size_t i = 0; i < 8; i++)
            sla.share_credits[i] = 127;
386
        desc_complete_indir(0, &sla, sizeof(sla));
387
    } else if (d->opcode == i40e_aqc_opc_remove_macvlan) {
388
389
390
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  remove macvlan" << std::endl;
#endif
391
392
393
394
395
396
397
398
        struct i40e_aqc_macvlan *m = reinterpret_cast<
            struct i40e_aqc_macvlan *>(d->params.raw);
        struct i40e_aqc_remove_macvlan_element_data *rve =
            reinterpret_cast<struct i40e_aqc_remove_macvlan_element_data *>(
                    data);
        for (uint16_t i = 0; i < m->num_addresses; i++)
            rve[i].error_code = I40E_AQC_REMOVE_MACVLAN_SUCCESS;

399
        desc_complete_indir(0, data, d->datalen);
400
    } else {
401
402
403
#ifdef DEBUG_ADMINQ
        std::cerr << "atq:  uknown opcode=" << d->opcode << std::endl;
#endif
404
405
        //desc_complete(I40E_AQ_RC_ESRCH);
        desc_complete(0);
406
407
    }
}