[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <56D066D7.2090708@gmail.com>
Date: Fri, 26 Feb 2016 06:53:11 -0800
From: John Fastabend <john.fastabend@...il.com>
To: Pablo Neira Ayuso <pablo@...filter.org>
CC: netdev@...r.kernel.org, davem@...emloft.net, jiri@...nulli.us,
horms@...ge.net.au
Subject: Re: [PATCH RFC 3/3] net: convert tc_u32 to use the intermediate representation
On 16-02-26 06:24 AM, Pablo Neira Ayuso wrote:
> On Thu, Feb 25, 2016 at 12:37:38PM -0800, John Fastabend wrote:
>> On 16-02-25 09:37 AM, Pablo Neira Ayuso wrote:
>>> This patch moves the u32 parser from the ixgbe that John has made to the
>>> core u32. This parser has been adapted to build the intermediate
>>> representation.
>>>
>>> To store the parsing information, this patch introduces a parse table
>>> object, one per device, so we don't need to store the parsing states in
>>> the adapter, which is the major dependency with previous patches.
>>>
>>> Since u32 allows rules connected via links, the u32 parser tracks this
>>> links and then generates the intermediate representation that is passed
>>> to the ixgbe driver.
>>
>> It also supports hash tables and loops.
>
> Then, it is a matter of extending the tc-u32 parser and the IR to
> support whatever you need.
sure but its easier for me to just consume u32 at the moment. And I
am concerned about performance overhead with this IR. I'm going to need
to have a performant solution eventually and I don't like converting
into one IR only to go into another.
>
>>> New drivers will only have to implement the jit translation code based
>>> on the intermediate representation. With some extra work, I think it
>>> should be possible to generalize the existing tc specific ndo action so
>>> it can be used by other frontends.
>>>
>>> I tried to stick to John's original u32 frontend parser as much as
>>> possible, adapting it to build the intermediate representation.
>>>
>>> After this change, we don't expose the tc action structure layout and
>>> other similar frontend details to the backend anymore to the backend
>>> anymore. I think this is good since changes in the frontend should not
>>> need to be propagated to the 1..n drivers supporting u32 offloads. In
>>> that sense, this helps to keep the frontend software representation
>>> separated from low-level backend driver details.
>>>
>>> After this patch, it should be possible to put the tc_cls_u32_knode
>>> structure into diet since we only need the handle (as unique id) and the
>>> ast tree.
>>>
>>
>> On ixgbe this is true but going forward I can support hash functions
>> so you need the divisor, prio, and handle minimally. I'm not sure how
>> to do hash tables for example in this IR yet. Might be there I'm still
>> trying to grok the IR details.
>
> We just need a description with the list elements that you want to
> place in the hashtable.
>
> The matching can be expressed in ast by adding a list of elements that
> are part of the hashtable:
>
> relational
> / \
> / \
> / \
> payload list of elements
> (offset, base, len) |
> \
> --> e1 --> e2 --> ... --> en
Sure it can be done but taking the u32 directly and flower directly is
so easy I don't see the need for the complexity.
>
>>> I couldn't send any more incremental changes to update previous work
>>> since the u32 parser and the internal representation were put together,
>>> that why this patch is slightly large.
>>>
>>> Signed-off-by: Pablo Neira Ayuso <pablo@...filter.org>
>>> ---
>>> drivers/net/ethernet/intel/ixgbe/ixgbe.h | 4 -
>>> drivers/net/ethernet/intel/ixgbe/ixgbe_main.c | 216 ++++++++--------
>>> drivers/net/ethernet/intel/ixgbe/ixgbe_model.h | 112 --------
>>> include/net/pkt_cls.h | 3 +
>>> net/sched/cls_u32.c | 344 +++++++++++++++++++++++++
>>
>> Feels like a lot of code to me when the original direct u32 to hw
>> was reasonably small and the flower implementation is small as well.
>
> This is your tc-u32 parser plus simple boiler plate code to build the
> IR that will be common to everyone. Without this, other drivers will
> have to track u32 links and so on.
>
But you lost the model piece which allows me to easily change the
hardware underneath the driver.
> Other than that, we'll end up with 1..N different u32 parsers in the
> backend.
>
well if folks like the way I wrote the u32 parser we could standardize
the model and header for u32 and give them helper functions. But by
writing your own model you can optimize it. Anyways at the moment it
looks like we will only have two the Intel and mlx ones I haven't
seen anyone else working on this yet.
>> [...]
>>
>>>
>>> +static int ixgbe_tcp_jit(struct net_ir_jit_ctx *ctx,
>>> + const struct net_ir_expr *expr,
>>> + void *data)
>>> +{
>>> + struct ixgbe_filter *f = (struct ixgbe_filter *)data;
>>> + struct net_ir_expr *right = expr->relational.right;
>>> + struct net_ir_expr *payload;
>>> + u32 mask = 0xffffffff;
>>> +
>>> + if (expr->relational.left->type == NET_IR_EXPR_BINOP) {
>>> + payload = expr->relational.left->binop.left;
>>> + mask = expr->relational.left->binop.right->value.data;
>>> + } else {
>>> + payload = expr->relational.left;
>>> + }
>>> +
>>> + switch (payload->payload.offset) {
>>
>> I don't like this because it hardcodes the offset pieces into the
>> driver. The reason I used a model.h header file with its structures
>> in ixgbe is the next step is to expose the parser so the model can be
>> updated. So that when the hardware parser changes the data structure
>> can be changed via firmware call. The style of coding here wont work
>> for that so we still need the model.h file and for loop.
>
> Model update via firmware?
>
> I think it's been already several conference rounds insisting on the
> fact that we don't want infrastructure that allows to insert binary
> blobs/sdks.
Its a firmware/ucode update we do this already today and the
infrastructure to do it has been in the kernel for a long time.
Firmware has always been a binary blob it has to be. All this means is
we have to read the parser out at init() time of the driver instead of
using a static model.h header. Hardware is not going to be static
anymore.
Even if its based on the device ID its the same problem I wouldn't
want a bunch of if (device X) do this else (device y) do that and
so on.
>
> I don't think this is the way to go.
>
>> [...]
>>
>>> static int ixgbe_configure_clsu32(struct ixgbe_adapter *adapter,
>>> __be16 protocol,
>>> struct tc_cls_u32_offload *cls)
>>> {
>>> u32 loc = cls->knode.handle & 0xfffff;
>>> struct ixgbe_hw *hw = &adapter->hw;
>>> - struct ixgbe_mat_field *field_ptr;
>>> struct ixgbe_filter f;
>>> -#ifdef CONFIG_NET_CLS_ACT
>>> - const struct tc_action *a;
>>> -#endif
>>> - int i, err = 0;
>>> + int err = 0;
>>> u32 handle;
>>>
>>> memset(&f.mask, 0, sizeof(union ixgbe_atr_input));
>>> handle = cls->knode.handle;
>>>
>>> - /* At the moment cls_u32 jumps to transport layer and skips past
>>> - * L2 headers. The canonical method to match L2 frames is to use
>>> - * negative values. However this is error prone at best but really
>>> - * just broken because there is no way to "know" what sort of hdr
>>> - * is in front of the transport layer. Fix cls_u32 to support L2
>>> - * headers when needed.
>>> - */
>>> - if (protocol != htons(ETH_P_IP))
>>
>> This was hardcoded originally just to find the inital offset going
>> forward I want to patch u32 to start at the link layer to avoid
>> this.
>
> Then, you extend the tc-u32 parser, but from the frontend core
> infrastructure code, so everyone we'll benefit for this incremental
> update.
>
We will push these patches into u32 anyways and it doesn't need the IR
to benefit everyone.
>> [...]
>>
>>> - if (!found_entry)
>>> - goto err_out;
>>> - }
>>> + if (net_ir_jit(&cls->knode.ast, &ixgbe_desc, &f) < 0)
>>> + return -EINVAL;
>>>
>>
>> OK but this takes a cls_u32 offload engine that was running on a handful
>> of LOC + a header file description that in the future will be read out
>> of firmware and replaces it with an entire infrastructure. I'm not
>> convinced the added complexity is worth it.
>
> It's all about consolidating infrastructure.
>
sure I just think we are getting ahead of ourselves here and
consolidating a single driver and single classifier and creating a lot
of code along the way when the current implementation is simple IMO with
this I have to follow an AST and pointers and a bunch of allocs. As the
guy writing the driver I see no need for it yet.
>> [...]
>>
>>> --- a/drivers/net/ethernet/intel/ixgbe/ixgbe_model.h
>>> +++ /dev/null
>>> @@ -1,112 +0,0 @@
>>> -/*******************************************************************************
>>> - *
>>> - * Intel 10 Gigabit PCI Express Linux drive
>>> - * Copyright(c) 2016 Intel Corporation.
>>> - *
>>> - * This program is free software; you can redistribute it and/or modify it
>>> - * under the terms and conditions of the GNU General Public License,
>>> - * version 2, as published by the Free Software Foundation.
>>> - *
>>> - * This program is distributed in the hope it will be useful, but WITHOUT
>>> - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
>>> - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
>>> - * more details.
>>> - *
>>> - * You should have received a copy of the GNU General Public License along
>>> - * with this program. If not, see <http://www.gnu.org/licenses/>.
>>> - *
>>> - * The full GNU General Public License is included in this distribution in
>>> - * the file called "COPYING".
>>> - *
>>> - * Contact Information:
>>> - * e1000-devel Mailing List <e1000-devel@...ts.sourceforge.net>
>>> - * Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
>>> - *
>>> - ******************************************************************************/
>>> -
>>> -#ifndef _IXGBE_MODEL_H_
>>> -#define _IXGBE_MODEL_H_
>>> -
>>> -#include "ixgbe.h"
>>> -#include "ixgbe_type.h"
>>> -
>>> -struct ixgbe_mat_field {
>>> - unsigned int off;
>>> - unsigned int mask;
>>> - int (*val)(struct ixgbe_fdir_filter *input,
>>> - union ixgbe_atr_input *mask,
>>> - u32 val, u32 m);
>>> - unsigned int type;
>>> -};
>>
>> As noted above the intent is to make the parser programmable so
>> something like this model needs to exist for firmware to populate.
>
> Oh, again references to update via firmware.
>
> [...]
>> My preference is to hold this work until we get a few more driver
>> implementation for multiple things and see if that can scale easily
>> as I suspect it might without having to over-engineer it.
>
> I don't agree, that's simply preventing by now other frontends and
> drivers to get offload support.
>
I don't think so at all. The driver support is simple at the moment
and the interface is still evolving in the frontend. What is this
preventing now? I can submit my flower implementation but the mlx folks
wanted to do it. The other concern about performance is also a big
one for me. And the AST as it is doesn't solve my eBPF case either so
maybe we should just lower everything to eBPF and have one IR instead
of two if we really want a universal IR.
.John
Powered by blists - more mailing lists