#include <linux/ptp_clock_kernel.h>
 #include <linux/hrtimer.h>
 #include <linux/debugfs.h>
+#include <net/flow_offload.h>
 
 #include "sparx5_main_regs.h"
 
        u16 pgid_idx;
 };
 
+struct sparx5_mall_entry {
+       struct list_head list;
+       struct sparx5_port *port;
+       unsigned long cookie;
+       enum flow_action_id type;
+       bool ingress;
+};
+
 #define SPARX5_PTP_TIMEOUT             msecs_to_jiffies(10)
 #define SPARX5_SKB_CB(skb) \
        ((struct sparx5_skb_cb *)((skb)->cb))
        struct vcap_control *vcap_ctrl;
        /* PGID allocation map */
        u8 pgid_map[PGID_TABLE_SIZE];
+       struct list_head mall_entries;
        /* Common root for debugfs */
        struct dentry *debugfs_root;
 };
 
 #include "sparx5_main.h"
 #include "sparx5_vcap_impl.h"
 
+static struct sparx5_mall_entry *
+sparx5_tc_matchall_entry_find(struct list_head *entries, unsigned long cookie)
+{
+       struct sparx5_mall_entry *entry;
+
+       list_for_each_entry(entry, entries, list) {
+               if (entry->cookie == cookie)
+                       return entry;
+       }
+
+       return NULL;
+}
+
+static void sparx5_tc_matchall_parse_action(struct sparx5_port *port,
+                                           struct sparx5_mall_entry *entry,
+                                           struct flow_action_entry *action,
+                                           bool ingress,
+                                           unsigned long cookie)
+{
+       entry->port = port;
+       entry->type = action->id;
+       entry->ingress = ingress;
+       entry->cookie = cookie;
+}
+
 static int sparx5_tc_matchall_replace(struct net_device *ndev,
                                      struct tc_cls_matchall_offload *tmo,
                                      bool ingress)
 {
        struct sparx5_port *port = netdev_priv(ndev);
+       struct sparx5_mall_entry *mall_entry;
        struct flow_action_entry *action;
        struct sparx5 *sparx5;
        int err;
        }
        action = &tmo->rule->action.entries[0];
 
+       mall_entry = kzalloc(sizeof(*mall_entry), GFP_KERNEL);
+       if (!mall_entry)
+               return -ENOMEM;
+
+       sparx5_tc_matchall_parse_action(port,
+                                       mall_entry,
+                                       action,
+                                       ingress,
+                                       tmo->cookie);
+
        sparx5 = port->sparx5;
        switch (action->id) {
        case FLOW_ACTION_GOTO:
                NL_SET_ERR_MSG_MOD(tmo->common.extack, "Unsupported action");
                return -EOPNOTSUPP;
        }
+
+       list_add_tail(&mall_entry->list, &sparx5->mall_entries);
+
        return 0;
 }
 
                                      bool ingress)
 {
        struct sparx5_port *port = netdev_priv(ndev);
-       struct sparx5 *sparx5;
+       struct sparx5 *sparx5 = port->sparx5;
+       struct sparx5_mall_entry *entry;
        int err;
 
-       sparx5 = port->sparx5;
-       if (!tmo->rule && tmo->cookie) {
+       entry = sparx5_tc_matchall_entry_find(&sparx5->mall_entries,
+                                             tmo->cookie);
+       if (!entry)
+               return -ENOENT;
+
+       if (entry->type == FLOW_ACTION_GOTO) {
                err = vcap_enable_lookups(sparx5->vcap_ctrl, ndev,
                                          0, 0, tmo->cookie, false);
-               if (err)
-                       return err;
-               return 0;
+       } else {
+               NL_SET_ERR_MSG_MOD(tmo->common.extack, "Unsupported action");
+               err = -EOPNOTSUPP;
        }
-       NL_SET_ERR_MSG_MOD(tmo->common.extack, "Unsupported action");
-       return -EOPNOTSUPP;
+
+       list_del(&entry->list);
+
+       return err;
 }
 
 int sparx5_tc_matchall(struct net_device *ndev,