mirror of
https://github.com/coolsnowwolf/lede.git
synced 2025-04-16 04:13:31 +00:00
generic: sync kernel 6.6 from upstream
This commit is contained in:
parent
bc21d2c7d6
commit
ce6d48a35d
@ -1,75 +0,0 @@
|
||||
--- a/init/Kconfig
|
||||
+++ b/init/Kconfig
|
||||
@@ -1356,16 +1356,6 @@ config BOOT_CONFIG_EMBED_FILE
|
||||
This bootconfig will be used if there is no initrd or no other
|
||||
bootconfig in the initrd.
|
||||
|
||||
-config INITRAMFS_PRESERVE_MTIME
|
||||
- bool "Preserve cpio archive mtimes in initramfs"
|
||||
- default y
|
||||
- help
|
||||
- Each entry in an initramfs cpio archive carries an mtime value. When
|
||||
- enabled, extracted cpio items take this mtime, with directory mtime
|
||||
- setting deferred until after creation of any child entries.
|
||||
-
|
||||
- If unsure, say Y.
|
||||
-
|
||||
choice
|
||||
prompt "Compiler optimization level"
|
||||
default CC_OPTIMIZE_FOR_PERFORMANCE
|
||||
--- a/init/initramfs.c
|
||||
+++ b/init/initramfs.c
|
||||
@@ -121,17 +121,15 @@ static void __init free_hash(void)
|
||||
}
|
||||
}
|
||||
|
||||
-#ifdef CONFIG_INITRAMFS_PRESERVE_MTIME
|
||||
-static void __init do_utime(char *filename, time64_t mtime)
|
||||
+static long __init do_utime(char *filename, time64_t mtime)
|
||||
{
|
||||
- struct timespec64 t[2] = { { .tv_sec = mtime }, { .tv_sec = mtime } };
|
||||
- init_utimes(filename, t);
|
||||
-}
|
||||
+ struct timespec64 t[2];
|
||||
|
||||
-static void __init do_utime_path(const struct path *path, time64_t mtime)
|
||||
-{
|
||||
- struct timespec64 t[2] = { { .tv_sec = mtime }, { .tv_sec = mtime } };
|
||||
- vfs_utimes(path, t);
|
||||
+ t[0].tv_sec = mtime;
|
||||
+ t[0].tv_nsec = 0;
|
||||
+ t[1].tv_sec = mtime;
|
||||
+ t[1].tv_nsec = 0;
|
||||
+ return init_utimes(filename, t);
|
||||
}
|
||||
|
||||
static __initdata LIST_HEAD(dir_list);
|
||||
@@ -164,12 +162,6 @@ static void __init dir_utime(void)
|
||||
kfree(de);
|
||||
}
|
||||
}
|
||||
-#else
|
||||
-static void __init do_utime(char *filename, time64_t mtime) {}
|
||||
-static void __init do_utime_path(const struct path *path, time64_t mtime) {}
|
||||
-static void __init dir_add(const char *name, time64_t mtime) {}
|
||||
-static void __init dir_utime(void) {}
|
||||
-#endif
|
||||
|
||||
static __initdata time64_t mtime;
|
||||
|
||||
@@ -401,10 +393,14 @@ static int __init do_name(void)
|
||||
static int __init do_copy(void)
|
||||
{
|
||||
if (byte_count >= body_len) {
|
||||
+ struct timespec64 t[2] = { };
|
||||
if (xwrite(wfile, victim, body_len, &wfile_pos) != body_len)
|
||||
error("write error");
|
||||
|
||||
- do_utime_path(&wfile->f_path, mtime);
|
||||
+ t[0].tv_sec = mtime;
|
||||
+ t[1].tv_sec = mtime;
|
||||
+ vfs_utimes(&wfile->f_path, t);
|
||||
+
|
||||
fput(wfile);
|
||||
if (csum_present && io_csum != hdr_csum)
|
||||
error("bad data checksum");
|
@ -1,146 +0,0 @@
|
||||
--- a/include/linux/netdevice.h
|
||||
+++ b/include/linux/netdevice.h
|
||||
@@ -576,8 +576,8 @@ static inline bool napi_if_scheduled_mar
|
||||
{
|
||||
unsigned long val, new;
|
||||
|
||||
- val = READ_ONCE(n->state);
|
||||
do {
|
||||
+ val = READ_ONCE(n->state);
|
||||
if (val & NAPIF_STATE_DISABLE)
|
||||
return true;
|
||||
|
||||
@@ -585,7 +585,7 @@ static inline bool napi_if_scheduled_mar
|
||||
return false;
|
||||
|
||||
new = val | NAPIF_STATE_MISSED;
|
||||
- } while (!try_cmpxchg(&n->state, &val, new));
|
||||
+ } while (cmpxchg(&n->state, val, new) != val);
|
||||
|
||||
return true;
|
||||
}
|
||||
@@ -1906,6 +1906,7 @@ enum netdev_stat_type {
|
||||
* @tipc_ptr: TIPC specific data
|
||||
* @atalk_ptr: AppleTalk link
|
||||
* @ip_ptr: IPv4 specific data
|
||||
+ * @dn_ptr: DECnet specific data
|
||||
* @ip6_ptr: IPv6 specific data
|
||||
* @ax25_ptr: AX.25 specific data
|
||||
* @ieee80211_ptr: IEEE 802.11 specific data, assign before registering
|
||||
@@ -2202,8 +2203,6 @@ struct net_device {
|
||||
|
||||
/* Protocol-specific pointers */
|
||||
|
||||
- struct in_device __rcu *ip_ptr;
|
||||
- struct inet6_dev __rcu *ip6_ptr;
|
||||
#if IS_ENABLED(CONFIG_VLAN_8021Q)
|
||||
struct vlan_info __rcu *vlan_info;
|
||||
#endif
|
||||
@@ -2216,15 +2215,16 @@ struct net_device {
|
||||
#if IS_ENABLED(CONFIG_ATALK)
|
||||
void *atalk_ptr;
|
||||
#endif
|
||||
+ struct in_device __rcu *ip_ptr;
|
||||
+#if IS_ENABLED(CONFIG_DECNET)
|
||||
+ struct dn_dev __rcu *dn_ptr;
|
||||
+#endif
|
||||
+ struct inet6_dev __rcu *ip6_ptr;
|
||||
#if IS_ENABLED(CONFIG_AX25)
|
||||
void *ax25_ptr;
|
||||
#endif
|
||||
-#if IS_ENABLED(CONFIG_CFG80211)
|
||||
struct wireless_dev *ieee80211_ptr;
|
||||
-#endif
|
||||
-#if IS_ENABLED(CONFIG_IEEE802154) || IS_ENABLED(CONFIG_6LOWPAN)
|
||||
struct wpan_dev *ieee802154_ptr;
|
||||
-#endif
|
||||
#if IS_ENABLED(CONFIG_MPLS_ROUTING)
|
||||
struct mpls_dev __rcu *mpls_ptr;
|
||||
#endif
|
||||
@@ -2655,6 +2655,8 @@ netif_napi_add_tx_weight(struct net_devi
|
||||
netif_napi_add_weight(dev, napi, poll, weight);
|
||||
}
|
||||
|
||||
+#define netif_tx_napi_add netif_napi_add_tx_weight
|
||||
+
|
||||
/**
|
||||
* netif_napi_add_tx() - initialize a NAPI context to be used for Tx only
|
||||
* @dev: network device
|
||||
--- a/include/net/cfg802154.h
|
||||
+++ b/include/net/cfg802154.h
|
||||
@@ -482,7 +482,6 @@ struct wpan_dev {
|
||||
|
||||
#define to_phy(_dev) container_of(_dev, struct wpan_phy, dev)
|
||||
|
||||
-#if IS_ENABLED(CONFIG_IEEE802154) || IS_ENABLED(CONFIG_6LOWPAN)
|
||||
static inline int
|
||||
wpan_dev_hard_header(struct sk_buff *skb, struct net_device *dev,
|
||||
const struct ieee802154_addr *daddr,
|
||||
@@ -493,7 +492,6 @@ wpan_dev_hard_header(struct sk_buff *skb
|
||||
|
||||
return wpan_dev->header_ops->create(skb, dev, daddr, saddr, len);
|
||||
}
|
||||
-#endif
|
||||
|
||||
struct wpan_phy *
|
||||
wpan_phy_new(const struct cfg802154_ops *ops, size_t priv_size);
|
||||
--- a/net/batman-adv/hard-interface.c
|
||||
+++ b/net/batman-adv/hard-interface.c
|
||||
@@ -309,11 +309,9 @@ static bool batadv_is_cfg80211_netdev(st
|
||||
if (!net_device)
|
||||
return false;
|
||||
|
||||
-#if IS_ENABLED(CONFIG_CFG80211)
|
||||
/* cfg80211 drivers have to set ieee80211_ptr */
|
||||
if (net_device->ieee80211_ptr)
|
||||
return true;
|
||||
-#endif
|
||||
|
||||
return false;
|
||||
}
|
||||
--- a/net/core/net-sysfs.c
|
||||
+++ b/net/core/net-sysfs.c
|
||||
@@ -748,6 +748,7 @@ static const struct attribute_group nets
|
||||
.attrs = netstat_attrs,
|
||||
};
|
||||
|
||||
+#if IS_ENABLED(CONFIG_WIRELESS_EXT) || IS_ENABLED(CONFIG_CFG80211)
|
||||
static struct attribute *wireless_attrs[] = {
|
||||
NULL
|
||||
};
|
||||
@@ -756,19 +757,7 @@ static const struct attribute_group wire
|
||||
.name = "wireless",
|
||||
.attrs = wireless_attrs,
|
||||
};
|
||||
-
|
||||
-static bool wireless_group_needed(struct net_device *ndev)
|
||||
-{
|
||||
-#if IS_ENABLED(CONFIG_CFG80211)
|
||||
- if (ndev->ieee80211_ptr)
|
||||
- return true;
|
||||
#endif
|
||||
-#if IS_ENABLED(CONFIG_WIRELESS_EXT)
|
||||
- if (ndev->wireless_handlers)
|
||||
- return true;
|
||||
-#endif
|
||||
- return false;
|
||||
-}
|
||||
|
||||
#else /* CONFIG_SYSFS */
|
||||
#define net_class_groups NULL
|
||||
@@ -2037,8 +2026,14 @@ int netdev_register_kobject(struct net_d
|
||||
|
||||
*groups++ = &netstat_group;
|
||||
|
||||
- if (wireless_group_needed(ndev))
|
||||
+#if IS_ENABLED(CONFIG_WIRELESS_EXT) || IS_ENABLED(CONFIG_CFG80211)
|
||||
+ if (ndev->ieee80211_ptr)
|
||||
+ *groups++ = &wireless_group;
|
||||
+#if IS_ENABLED(CONFIG_WIRELESS_EXT)
|
||||
+ else if (ndev->wireless_handlers)
|
||||
*groups++ = &wireless_group;
|
||||
+#endif
|
||||
+#endif
|
||||
#endif /* CONFIG_SYSFS */
|
||||
|
||||
error = device_add(dev);
|
@ -1,279 +0,0 @@
|
||||
From ca31fb1ed58c293b3f02b1aa46aa672866aff540 Mon Sep 17 00:00:00 2001
|
||||
From: Marty Jones <mj8263788@gmail.com>
|
||||
Date: Thu, 17 Aug 2023 17:06:55 -0400
|
||||
Subject: [PATCH 7/8] Revert "genetlink: remove userhdr from struct genl_info"
|
||||
|
||||
This reverts commit bffcc6882a1bb2be8c9420184966f4c2c822078e.
|
||||
---
|
||||
drivers/block/drbd/drbd_nl.c | 9 ++++-----
|
||||
include/net/genetlink.h | 7 ++-----
|
||||
net/netlink/genetlink.c | 1 +
|
||||
net/openvswitch/conntrack.c | 2 +-
|
||||
net/openvswitch/datapath.c | 29 +++++++++++++----------------
|
||||
net/openvswitch/meter.c | 10 +++++-----
|
||||
net/tipc/netlink_compat.c | 2 +-
|
||||
7 files changed, 27 insertions(+), 33 deletions(-)
|
||||
|
||||
--- a/drivers/block/drbd/drbd_nl.c
|
||||
+++ b/drivers/block/drbd/drbd_nl.c
|
||||
@@ -159,7 +159,7 @@ static int drbd_msg_sprintf_info(struct
|
||||
static int drbd_adm_prepare(struct drbd_config_context *adm_ctx,
|
||||
struct sk_buff *skb, struct genl_info *info, unsigned flags)
|
||||
{
|
||||
- struct drbd_genlmsghdr *d_in = genl_info_userhdr(info);
|
||||
+ struct drbd_genlmsghdr *d_in = info->userhdr;
|
||||
const u8 cmd = info->genlhdr->cmd;
|
||||
int err;
|
||||
|
||||
@@ -1396,9 +1396,8 @@ static void drbd_suspend_al(struct drbd_
|
||||
|
||||
static bool should_set_defaults(struct genl_info *info)
|
||||
{
|
||||
- struct drbd_genlmsghdr *dh = genl_info_userhdr(info);
|
||||
-
|
||||
- return 0 != (dh->flags & DRBD_GENL_F_SET_DEFAULTS);
|
||||
+ unsigned flags = ((struct drbd_genlmsghdr*)info->userhdr)->flags;
|
||||
+ return 0 != (flags & DRBD_GENL_F_SET_DEFAULTS);
|
||||
}
|
||||
|
||||
static unsigned int drbd_al_extents_max(struct drbd_backing_dev *bdev)
|
||||
@@ -4277,7 +4276,7 @@ static void device_to_info(struct device
|
||||
int drbd_adm_new_minor(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct drbd_config_context adm_ctx;
|
||||
- struct drbd_genlmsghdr *dh = genl_info_userhdr(info);
|
||||
+ struct drbd_genlmsghdr *dh = info->userhdr;
|
||||
enum drbd_ret_code retcode;
|
||||
|
||||
retcode = drbd_adm_prepare(&adm_ctx, skb, info, DRBD_ADM_NEED_RESOURCE);
|
||||
--- a/include/net/genetlink.h
|
||||
+++ b/include/net/genetlink.h
|
||||
@@ -98,6 +98,7 @@ struct genl_family {
|
||||
* @family: generic netlink family
|
||||
* @nlhdr: netlink message header
|
||||
* @genlhdr: generic netlink message header
|
||||
+ * @userhdr: user specific header
|
||||
* @attrs: netlink attributes
|
||||
* @_net: network namespace
|
||||
* @user_ptr: user pointers
|
||||
@@ -109,6 +110,7 @@ struct genl_info {
|
||||
const struct genl_family *family;
|
||||
const struct nlmsghdr * nlhdr;
|
||||
struct genlmsghdr * genlhdr;
|
||||
+ void * userhdr;
|
||||
struct nlattr ** attrs;
|
||||
possible_net_t _net;
|
||||
void * user_ptr[2];
|
||||
@@ -125,11 +127,6 @@ static inline void genl_info_net_set(str
|
||||
write_pnet(&info->_net, net);
|
||||
}
|
||||
|
||||
-static inline void *genl_info_userhdr(const struct genl_info *info)
|
||||
-{
|
||||
- return (u8 *)info->genlhdr + GENL_HDRLEN;
|
||||
-}
|
||||
-
|
||||
#define GENL_SET_ERR_MSG(info, msg) NL_SET_ERR_MSG((info)->extack, msg)
|
||||
|
||||
#define GENL_SET_ERR_MSG_FMT(info, msg, args...) \
|
||||
--- a/net/netlink/genetlink.c
|
||||
+++ b/net/netlink/genetlink.c
|
||||
@@ -957,6 +957,7 @@ static int genl_family_rcv_msg_doit(cons
|
||||
info.family = family;
|
||||
info.nlhdr = nlh;
|
||||
info.genlhdr = nlmsg_data(nlh);
|
||||
+ info.userhdr = nlmsg_data(nlh) + GENL_HDRLEN;
|
||||
info.attrs = attrbuf;
|
||||
info.extack = extack;
|
||||
genl_info_net_set(&info, net);
|
||||
--- a/net/openvswitch/conntrack.c
|
||||
+++ b/net/openvswitch/conntrack.c
|
||||
@@ -1605,7 +1605,7 @@ static struct sk_buff *
|
||||
ovs_ct_limit_cmd_reply_start(struct genl_info *info, u8 cmd,
|
||||
struct ovs_header **ovs_reply_header)
|
||||
{
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct sk_buff *skb;
|
||||
|
||||
skb = genlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
|
||||
--- a/net/openvswitch/datapath.c
|
||||
+++ b/net/openvswitch/datapath.c
|
||||
@@ -590,7 +590,7 @@ out:
|
||||
|
||||
static int ovs_packet_cmd_execute(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct net *net = sock_net(skb->sk);
|
||||
struct nlattr **a = info->attrs;
|
||||
struct sw_flow_actions *acts;
|
||||
@@ -967,7 +967,7 @@ static int ovs_flow_cmd_new(struct sk_bu
|
||||
{
|
||||
struct net *net = sock_net(skb->sk);
|
||||
struct nlattr **a = info->attrs;
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct sw_flow *flow = NULL, *new_flow;
|
||||
struct sw_flow_mask mask;
|
||||
struct sk_buff *reply;
|
||||
@@ -1214,7 +1214,7 @@ static int ovs_flow_cmd_set(struct sk_bu
|
||||
{
|
||||
struct net *net = sock_net(skb->sk);
|
||||
struct nlattr **a = info->attrs;
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct sw_flow_key key;
|
||||
struct sw_flow *flow;
|
||||
struct sk_buff *reply = NULL;
|
||||
@@ -1315,7 +1315,7 @@ error:
|
||||
static int ovs_flow_cmd_get(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct nlattr **a = info->attrs;
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct net *net = sock_net(skb->sk);
|
||||
struct sw_flow_key key;
|
||||
struct sk_buff *reply;
|
||||
@@ -1374,7 +1374,7 @@ unlock:
|
||||
static int ovs_flow_cmd_del(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct nlattr **a = info->attrs;
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct net *net = sock_net(skb->sk);
|
||||
struct sw_flow_key key;
|
||||
struct sk_buff *reply;
|
||||
@@ -1642,7 +1642,7 @@ static void ovs_dp_reset_user_features(s
|
||||
{
|
||||
struct datapath *dp;
|
||||
|
||||
- dp = lookup_datapath(sock_net(skb->sk), genl_info_userhdr(info),
|
||||
+ dp = lookup_datapath(sock_net(skb->sk), info->userhdr,
|
||||
info->attrs);
|
||||
if (IS_ERR(dp))
|
||||
return;
|
||||
@@ -1935,8 +1935,7 @@ static int ovs_dp_cmd_del(struct sk_buff
|
||||
return -ENOMEM;
|
||||
|
||||
ovs_lock();
|
||||
- dp = lookup_datapath(sock_net(skb->sk), genl_info_userhdr(info),
|
||||
- info->attrs);
|
||||
+ dp = lookup_datapath(sock_net(skb->sk), info->userhdr, info->attrs);
|
||||
err = PTR_ERR(dp);
|
||||
if (IS_ERR(dp))
|
||||
goto err_unlock_free;
|
||||
@@ -1969,8 +1968,7 @@ static int ovs_dp_cmd_set(struct sk_buff
|
||||
return -ENOMEM;
|
||||
|
||||
ovs_lock();
|
||||
- dp = lookup_datapath(sock_net(skb->sk), genl_info_userhdr(info),
|
||||
- info->attrs);
|
||||
+ dp = lookup_datapath(sock_net(skb->sk), info->userhdr, info->attrs);
|
||||
err = PTR_ERR(dp);
|
||||
if (IS_ERR(dp))
|
||||
goto err_unlock_free;
|
||||
@@ -2005,8 +2003,7 @@ static int ovs_dp_cmd_get(struct sk_buff
|
||||
return -ENOMEM;
|
||||
|
||||
ovs_lock();
|
||||
- dp = lookup_datapath(sock_net(skb->sk), genl_info_userhdr(info),
|
||||
- info->attrs);
|
||||
+ dp = lookup_datapath(sock_net(skb->sk), info->userhdr, info->attrs);
|
||||
if (IS_ERR(dp)) {
|
||||
err = PTR_ERR(dp);
|
||||
goto err_unlock_free;
|
||||
@@ -2249,7 +2246,7 @@ static void ovs_update_headroom(struct d
|
||||
static int ovs_vport_cmd_new(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct nlattr **a = info->attrs;
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct vport_parms parms;
|
||||
struct sk_buff *reply;
|
||||
struct vport *vport;
|
||||
@@ -2351,7 +2348,7 @@ static int ovs_vport_cmd_set(struct sk_b
|
||||
return -ENOMEM;
|
||||
|
||||
ovs_lock();
|
||||
- vport = lookup_vport(sock_net(skb->sk), genl_info_userhdr(info), a);
|
||||
+ vport = lookup_vport(sock_net(skb->sk), info->userhdr, a);
|
||||
err = PTR_ERR(vport);
|
||||
if (IS_ERR(vport))
|
||||
goto exit_unlock_free;
|
||||
@@ -2407,7 +2404,7 @@ static int ovs_vport_cmd_del(struct sk_b
|
||||
return -ENOMEM;
|
||||
|
||||
ovs_lock();
|
||||
- vport = lookup_vport(sock_net(skb->sk), genl_info_userhdr(info), a);
|
||||
+ vport = lookup_vport(sock_net(skb->sk), info->userhdr, a);
|
||||
err = PTR_ERR(vport);
|
||||
if (IS_ERR(vport))
|
||||
goto exit_unlock_free;
|
||||
@@ -2450,7 +2447,7 @@ exit_unlock_free:
|
||||
static int ovs_vport_cmd_get(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct nlattr **a = info->attrs;
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct sk_buff *reply;
|
||||
struct vport *vport;
|
||||
int err;
|
||||
--- a/net/openvswitch/meter.c
|
||||
+++ b/net/openvswitch/meter.c
|
||||
@@ -211,7 +211,7 @@ ovs_meter_cmd_reply_start(struct genl_in
|
||||
struct ovs_header **ovs_reply_header)
|
||||
{
|
||||
struct sk_buff *skb;
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
|
||||
skb = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_ATOMIC);
|
||||
if (!skb)
|
||||
@@ -272,7 +272,7 @@ error:
|
||||
|
||||
static int ovs_meter_cmd_features(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct ovs_header *ovs_reply_header;
|
||||
struct nlattr *nla, *band_nla;
|
||||
struct sk_buff *reply;
|
||||
@@ -409,7 +409,7 @@ static int ovs_meter_cmd_set(struct sk_b
|
||||
struct dp_meter *meter, *old_meter;
|
||||
struct sk_buff *reply;
|
||||
struct ovs_header *ovs_reply_header;
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct dp_meter_table *meter_tbl;
|
||||
struct datapath *dp;
|
||||
int err;
|
||||
@@ -482,7 +482,7 @@ exit_free_meter:
|
||||
|
||||
static int ovs_meter_cmd_get(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct ovs_header *ovs_reply_header;
|
||||
struct nlattr **a = info->attrs;
|
||||
struct dp_meter *meter;
|
||||
@@ -535,7 +535,7 @@ exit_unlock:
|
||||
|
||||
static int ovs_meter_cmd_del(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
- struct ovs_header *ovs_header = genl_info_userhdr(info);
|
||||
+ struct ovs_header *ovs_header = info->userhdr;
|
||||
struct ovs_header *ovs_reply_header;
|
||||
struct nlattr **a = info->attrs;
|
||||
struct dp_meter *old_meter;
|
||||
--- a/net/tipc/netlink_compat.c
|
||||
+++ b/net/tipc/netlink_compat.c
|
||||
@@ -1295,7 +1295,7 @@ static int tipc_nl_compat_recv(struct sk
|
||||
struct tipc_nl_compat_msg msg;
|
||||
struct nlmsghdr *req_nlh;
|
||||
struct nlmsghdr *rep_nlh;
|
||||
- struct tipc_genlmsghdr *req_userhdr = genl_info_userhdr(info);
|
||||
+ struct tipc_genlmsghdr *req_userhdr = info->userhdr;
|
||||
|
||||
memset(&msg, 0, sizeof(msg));
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,183 @@
|
||||
From e1fbfa4a995d42e02e22b0dff2f8b4fdee1504b3 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 14 Nov 2023 15:08:42 +0100
|
||||
Subject: [PATCH 2/3] net: phy: aquantia: move MMD_VEND define to header
|
||||
|
||||
Move MMD_VEND define to header to clean things up and in preparation for
|
||||
firmware loading support that require some define placed in
|
||||
aquantia_main.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/aquantia/aquantia.h | 69 +++++++++++++++++++++++
|
||||
drivers/net/phy/aquantia/aquantia_hwmon.c | 14 -----
|
||||
drivers/net/phy/aquantia/aquantia_main.c | 55 ------------------
|
||||
3 files changed, 69 insertions(+), 69 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/aquantia/aquantia.h
|
||||
+++ b/drivers/net/phy/aquantia/aquantia.h
|
||||
@@ -9,6 +9,75 @@
|
||||
#include <linux/device.h>
|
||||
#include <linux/phy.h>
|
||||
|
||||
+/* Vendor specific 1, MDIO_MMD_VEND1 */
|
||||
+#define VEND1_GLOBAL_FW_ID 0x0020
|
||||
+#define VEND1_GLOBAL_FW_ID_MAJOR GENMASK(15, 8)
|
||||
+#define VEND1_GLOBAL_FW_ID_MINOR GENMASK(7, 0)
|
||||
+
|
||||
+/* The following registers all have similar layouts; first the registers... */
|
||||
+#define VEND1_GLOBAL_CFG_10M 0x0310
|
||||
+#define VEND1_GLOBAL_CFG_100M 0x031b
|
||||
+#define VEND1_GLOBAL_CFG_1G 0x031c
|
||||
+#define VEND1_GLOBAL_CFG_2_5G 0x031d
|
||||
+#define VEND1_GLOBAL_CFG_5G 0x031e
|
||||
+#define VEND1_GLOBAL_CFG_10G 0x031f
|
||||
+/* ...and now the fields */
|
||||
+#define VEND1_GLOBAL_CFG_RATE_ADAPT GENMASK(8, 7)
|
||||
+#define VEND1_GLOBAL_CFG_RATE_ADAPT_NONE 0
|
||||
+#define VEND1_GLOBAL_CFG_RATE_ADAPT_USX 1
|
||||
+#define VEND1_GLOBAL_CFG_RATE_ADAPT_PAUSE 2
|
||||
+
|
||||
+/* Vendor specific 1, MDIO_MMD_VEND2 */
|
||||
+#define VEND1_THERMAL_PROV_HIGH_TEMP_FAIL 0xc421
|
||||
+#define VEND1_THERMAL_PROV_LOW_TEMP_FAIL 0xc422
|
||||
+#define VEND1_THERMAL_PROV_HIGH_TEMP_WARN 0xc423
|
||||
+#define VEND1_THERMAL_PROV_LOW_TEMP_WARN 0xc424
|
||||
+#define VEND1_THERMAL_STAT1 0xc820
|
||||
+#define VEND1_THERMAL_STAT2 0xc821
|
||||
+#define VEND1_THERMAL_STAT2_VALID BIT(0)
|
||||
+#define VEND1_GENERAL_STAT1 0xc830
|
||||
+#define VEND1_GENERAL_STAT1_HIGH_TEMP_FAIL BIT(14)
|
||||
+#define VEND1_GENERAL_STAT1_LOW_TEMP_FAIL BIT(13)
|
||||
+#define VEND1_GENERAL_STAT1_HIGH_TEMP_WARN BIT(12)
|
||||
+#define VEND1_GENERAL_STAT1_LOW_TEMP_WARN BIT(11)
|
||||
+
|
||||
+#define VEND1_GLOBAL_GEN_STAT2 0xc831
|
||||
+#define VEND1_GLOBAL_GEN_STAT2_OP_IN_PROG BIT(15)
|
||||
+
|
||||
+#define VEND1_GLOBAL_RSVD_STAT1 0xc885
|
||||
+#define VEND1_GLOBAL_RSVD_STAT1_FW_BUILD_ID GENMASK(7, 4)
|
||||
+#define VEND1_GLOBAL_RSVD_STAT1_PROV_ID GENMASK(3, 0)
|
||||
+
|
||||
+#define VEND1_GLOBAL_RSVD_STAT9 0xc88d
|
||||
+#define VEND1_GLOBAL_RSVD_STAT9_MODE GENMASK(7, 0)
|
||||
+#define VEND1_GLOBAL_RSVD_STAT9_1000BT2 0x23
|
||||
+
|
||||
+#define VEND1_GLOBAL_INT_STD_STATUS 0xfc00
|
||||
+#define VEND1_GLOBAL_INT_VEND_STATUS 0xfc01
|
||||
+
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK 0xff00
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_PMA1 BIT(15)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_PMA2 BIT(14)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_PCS1 BIT(13)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_PCS2 BIT(12)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_PCS3 BIT(11)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_PHY_XS1 BIT(10)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_PHY_XS2 BIT(9)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_AN1 BIT(8)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_AN2 BIT(7)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_GBE BIT(6)
|
||||
+#define VEND1_GLOBAL_INT_STD_MASK_ALL BIT(0)
|
||||
+
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK 0xff01
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK_PMA BIT(15)
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK_PCS BIT(14)
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK_PHY_XS BIT(13)
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK_AN BIT(12)
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK_GBE BIT(11)
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK_GLOBAL1 BIT(2)
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK_GLOBAL2 BIT(1)
|
||||
+#define VEND1_GLOBAL_INT_VEND_MASK_GLOBAL3 BIT(0)
|
||||
+
|
||||
#if IS_REACHABLE(CONFIG_HWMON)
|
||||
int aqr_hwmon_probe(struct phy_device *phydev);
|
||||
#else
|
||||
--- a/drivers/net/phy/aquantia/aquantia_hwmon.c
|
||||
+++ b/drivers/net/phy/aquantia/aquantia_hwmon.c
|
||||
@@ -13,20 +13,6 @@
|
||||
|
||||
#include "aquantia.h"
|
||||
|
||||
-/* Vendor specific 1, MDIO_MMD_VEND2 */
|
||||
-#define VEND1_THERMAL_PROV_HIGH_TEMP_FAIL 0xc421
|
||||
-#define VEND1_THERMAL_PROV_LOW_TEMP_FAIL 0xc422
|
||||
-#define VEND1_THERMAL_PROV_HIGH_TEMP_WARN 0xc423
|
||||
-#define VEND1_THERMAL_PROV_LOW_TEMP_WARN 0xc424
|
||||
-#define VEND1_THERMAL_STAT1 0xc820
|
||||
-#define VEND1_THERMAL_STAT2 0xc821
|
||||
-#define VEND1_THERMAL_STAT2_VALID BIT(0)
|
||||
-#define VEND1_GENERAL_STAT1 0xc830
|
||||
-#define VEND1_GENERAL_STAT1_HIGH_TEMP_FAIL BIT(14)
|
||||
-#define VEND1_GENERAL_STAT1_LOW_TEMP_FAIL BIT(13)
|
||||
-#define VEND1_GENERAL_STAT1_HIGH_TEMP_WARN BIT(12)
|
||||
-#define VEND1_GENERAL_STAT1_LOW_TEMP_WARN BIT(11)
|
||||
-
|
||||
#if IS_REACHABLE(CONFIG_HWMON)
|
||||
|
||||
static umode_t aqr_hwmon_is_visible(const void *data,
|
||||
--- a/drivers/net/phy/aquantia/aquantia_main.c
|
||||
+++ b/drivers/net/phy/aquantia/aquantia_main.c
|
||||
@@ -91,61 +91,6 @@
|
||||
#define MDIO_C22EXT_STAT_SGMII_TX_FRAME_ALIGN_ERR 0xd31a
|
||||
#define MDIO_C22EXT_STAT_SGMII_TX_RUNT_FRAMES 0xd31b
|
||||
|
||||
-/* Vendor specific 1, MDIO_MMD_VEND1 */
|
||||
-#define VEND1_GLOBAL_FW_ID 0x0020
|
||||
-#define VEND1_GLOBAL_FW_ID_MAJOR GENMASK(15, 8)
|
||||
-#define VEND1_GLOBAL_FW_ID_MINOR GENMASK(7, 0)
|
||||
-
|
||||
-#define VEND1_GLOBAL_GEN_STAT2 0xc831
|
||||
-#define VEND1_GLOBAL_GEN_STAT2_OP_IN_PROG BIT(15)
|
||||
-
|
||||
-/* The following registers all have similar layouts; first the registers... */
|
||||
-#define VEND1_GLOBAL_CFG_10M 0x0310
|
||||
-#define VEND1_GLOBAL_CFG_100M 0x031b
|
||||
-#define VEND1_GLOBAL_CFG_1G 0x031c
|
||||
-#define VEND1_GLOBAL_CFG_2_5G 0x031d
|
||||
-#define VEND1_GLOBAL_CFG_5G 0x031e
|
||||
-#define VEND1_GLOBAL_CFG_10G 0x031f
|
||||
-/* ...and now the fields */
|
||||
-#define VEND1_GLOBAL_CFG_RATE_ADAPT GENMASK(8, 7)
|
||||
-#define VEND1_GLOBAL_CFG_RATE_ADAPT_NONE 0
|
||||
-#define VEND1_GLOBAL_CFG_RATE_ADAPT_USX 1
|
||||
-#define VEND1_GLOBAL_CFG_RATE_ADAPT_PAUSE 2
|
||||
-
|
||||
-#define VEND1_GLOBAL_RSVD_STAT1 0xc885
|
||||
-#define VEND1_GLOBAL_RSVD_STAT1_FW_BUILD_ID GENMASK(7, 4)
|
||||
-#define VEND1_GLOBAL_RSVD_STAT1_PROV_ID GENMASK(3, 0)
|
||||
-
|
||||
-#define VEND1_GLOBAL_RSVD_STAT9 0xc88d
|
||||
-#define VEND1_GLOBAL_RSVD_STAT9_MODE GENMASK(7, 0)
|
||||
-#define VEND1_GLOBAL_RSVD_STAT9_1000BT2 0x23
|
||||
-
|
||||
-#define VEND1_GLOBAL_INT_STD_STATUS 0xfc00
|
||||
-#define VEND1_GLOBAL_INT_VEND_STATUS 0xfc01
|
||||
-
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK 0xff00
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_PMA1 BIT(15)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_PMA2 BIT(14)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_PCS1 BIT(13)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_PCS2 BIT(12)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_PCS3 BIT(11)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_PHY_XS1 BIT(10)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_PHY_XS2 BIT(9)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_AN1 BIT(8)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_AN2 BIT(7)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_GBE BIT(6)
|
||||
-#define VEND1_GLOBAL_INT_STD_MASK_ALL BIT(0)
|
||||
-
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK 0xff01
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK_PMA BIT(15)
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK_PCS BIT(14)
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK_PHY_XS BIT(13)
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK_AN BIT(12)
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK_GBE BIT(11)
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK_GLOBAL1 BIT(2)
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK_GLOBAL2 BIT(1)
|
||||
-#define VEND1_GLOBAL_INT_VEND_MASK_GLOBAL3 BIT(0)
|
||||
-
|
||||
/* Sleep and timeout for checking if the Processor-Intensive
|
||||
* MDIO operation is finished
|
||||
*/
|
@ -0,0 +1,504 @@
|
||||
From e93984ebc1c82bd34f7a1b3391efaceee0a8ae96 Mon Sep 17 00:00:00 2001
|
||||
From: Robert Marko <robimarko@gmail.com>
|
||||
Date: Tue, 14 Nov 2023 15:08:43 +0100
|
||||
Subject: [PATCH 3/3] net: phy: aquantia: add firmware load support
|
||||
|
||||
Aquantia PHY-s require firmware to be loaded before they start operating.
|
||||
It can be automatically loaded in case when there is a SPI-NOR connected
|
||||
to Aquantia PHY-s or can be loaded from the host via MDIO.
|
||||
|
||||
This patch adds support for loading the firmware via MDIO as in most cases
|
||||
there is no SPI-NOR being used to save on cost.
|
||||
Firmware loading code itself is ported from mainline U-boot with cleanups.
|
||||
|
||||
The firmware has mixed values both in big and little endian.
|
||||
PHY core itself is big-endian but it expects values to be in little-endian.
|
||||
The firmware is little-endian but CRC-16 value for it is stored at the end
|
||||
of firmware in big-endian.
|
||||
|
||||
It seems the PHY does the conversion internally from firmware that is
|
||||
little-endian to the PHY that is big-endian on using the mailbox
|
||||
but mailbox returns a big-endian CRC-16 to verify the written data
|
||||
integrity.
|
||||
|
||||
Co-developed-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: Robert Marko <robimarko@gmail.com>
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/aquantia/Kconfig | 1 +
|
||||
drivers/net/phy/aquantia/Makefile | 2 +-
|
||||
drivers/net/phy/aquantia/aquantia.h | 32 ++
|
||||
drivers/net/phy/aquantia/aquantia_firmware.c | 370 +++++++++++++++++++
|
||||
drivers/net/phy/aquantia/aquantia_main.c | 6 +
|
||||
5 files changed, 410 insertions(+), 1 deletion(-)
|
||||
create mode 100644 drivers/net/phy/aquantia/aquantia_firmware.c
|
||||
|
||||
--- a/drivers/net/phy/aquantia/Kconfig
|
||||
+++ b/drivers/net/phy/aquantia/Kconfig
|
||||
@@ -1,5 +1,6 @@
|
||||
# SPDX-License-Identifier: GPL-2.0-only
|
||||
config AQUANTIA_PHY
|
||||
tristate "Aquantia PHYs"
|
||||
+ select CRC_CCITT
|
||||
help
|
||||
Currently supports the Aquantia AQ1202, AQ2104, AQR105, AQR405
|
||||
--- a/drivers/net/phy/aquantia/Makefile
|
||||
+++ b/drivers/net/phy/aquantia/Makefile
|
||||
@@ -1,5 +1,5 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
-aquantia-objs += aquantia_main.o
|
||||
+aquantia-objs += aquantia_main.o aquantia_firmware.o
|
||||
ifdef CONFIG_HWMON
|
||||
aquantia-objs += aquantia_hwmon.o
|
||||
endif
|
||||
--- a/drivers/net/phy/aquantia/aquantia.h
|
||||
+++ b/drivers/net/phy/aquantia/aquantia.h
|
||||
@@ -10,10 +10,35 @@
|
||||
#include <linux/phy.h>
|
||||
|
||||
/* Vendor specific 1, MDIO_MMD_VEND1 */
|
||||
+#define VEND1_GLOBAL_SC 0x0
|
||||
+#define VEND1_GLOBAL_SC_SOFT_RESET BIT(15)
|
||||
+#define VEND1_GLOBAL_SC_LOW_POWER BIT(11)
|
||||
+
|
||||
#define VEND1_GLOBAL_FW_ID 0x0020
|
||||
#define VEND1_GLOBAL_FW_ID_MAJOR GENMASK(15, 8)
|
||||
#define VEND1_GLOBAL_FW_ID_MINOR GENMASK(7, 0)
|
||||
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE1 0x0200
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE1_EXECUTE BIT(15)
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE1_WRITE BIT(14)
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE1_CRC_RESET BIT(12)
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE1_BUSY BIT(8)
|
||||
+
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE2 0x0201
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE3 0x0202
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE3_MSW_ADDR_MASK GENMASK(15, 0)
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE3_MSW_ADDR(x) FIELD_PREP(VEND1_GLOBAL_MAILBOX_INTERFACE3_MSW_ADDR_MASK, (u16)((x) >> 16))
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE4 0x0203
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE4_LSW_ADDR_MASK GENMASK(15, 2)
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE4_LSW_ADDR(x) FIELD_PREP(VEND1_GLOBAL_MAILBOX_INTERFACE4_LSW_ADDR_MASK, (u16)(x))
|
||||
+
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE5 0x0204
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE5_MSW_DATA_MASK GENMASK(15, 0)
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE5_MSW_DATA(x) FIELD_PREP(VEND1_GLOBAL_MAILBOX_INTERFACE5_MSW_DATA_MASK, (u16)((x) >> 16))
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE6 0x0205
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE6_LSW_DATA_MASK GENMASK(15, 0)
|
||||
+#define VEND1_GLOBAL_MAILBOX_INTERFACE6_LSW_DATA(x) FIELD_PREP(VEND1_GLOBAL_MAILBOX_INTERFACE6_LSW_DATA_MASK, (u16)(x))
|
||||
+
|
||||
/* The following registers all have similar layouts; first the registers... */
|
||||
#define VEND1_GLOBAL_CFG_10M 0x0310
|
||||
#define VEND1_GLOBAL_CFG_100M 0x031b
|
||||
@@ -28,6 +53,11 @@
|
||||
#define VEND1_GLOBAL_CFG_RATE_ADAPT_PAUSE 2
|
||||
|
||||
/* Vendor specific 1, MDIO_MMD_VEND2 */
|
||||
+#define VEND1_GLOBAL_CONTROL2 0xc001
|
||||
+#define VEND1_GLOBAL_CONTROL2_UP_RUN_STALL_RST BIT(15)
|
||||
+#define VEND1_GLOBAL_CONTROL2_UP_RUN_STALL_OVD BIT(6)
|
||||
+#define VEND1_GLOBAL_CONTROL2_UP_RUN_STALL BIT(0)
|
||||
+
|
||||
#define VEND1_THERMAL_PROV_HIGH_TEMP_FAIL 0xc421
|
||||
#define VEND1_THERMAL_PROV_LOW_TEMP_FAIL 0xc422
|
||||
#define VEND1_THERMAL_PROV_HIGH_TEMP_WARN 0xc423
|
||||
@@ -83,3 +113,5 @@ int aqr_hwmon_probe(struct phy_device *p
|
||||
#else
|
||||
static inline int aqr_hwmon_probe(struct phy_device *phydev) { return 0; }
|
||||
#endif
|
||||
+
|
||||
+int aqr_firmware_load(struct phy_device *phydev);
|
||||
--- /dev/null
|
||||
+++ b/drivers/net/phy/aquantia/aquantia_firmware.c
|
||||
@@ -0,0 +1,370 @@
|
||||
+// SPDX-License-Identifier: GPL-2.0
|
||||
+
|
||||
+#include <linux/bitfield.h>
|
||||
+#include <linux/of.h>
|
||||
+#include <linux/firmware.h>
|
||||
+#include <linux/crc-ccitt.h>
|
||||
+#include <linux/nvmem-consumer.h>
|
||||
+
|
||||
+#include <asm/unaligned.h>
|
||||
+
|
||||
+#include "aquantia.h"
|
||||
+
|
||||
+#define UP_RESET_SLEEP 100
|
||||
+
|
||||
+/* addresses of memory segments in the phy */
|
||||
+#define DRAM_BASE_ADDR 0x3FFE0000
|
||||
+#define IRAM_BASE_ADDR 0x40000000
|
||||
+
|
||||
+/* firmware image format constants */
|
||||
+#define VERSION_STRING_SIZE 0x40
|
||||
+#define VERSION_STRING_OFFSET 0x0200
|
||||
+/* primary offset is written at an offset from the start of the fw blob */
|
||||
+#define PRIMARY_OFFSET_OFFSET 0x8
|
||||
+/* primary offset needs to be then added to a base offset */
|
||||
+#define PRIMARY_OFFSET_SHIFT 12
|
||||
+#define PRIMARY_OFFSET(x) ((x) << PRIMARY_OFFSET_SHIFT)
|
||||
+#define HEADER_OFFSET 0x300
|
||||
+
|
||||
+struct aqr_fw_header {
|
||||
+ u32 padding;
|
||||
+ u8 iram_offset[3];
|
||||
+ u8 iram_size[3];
|
||||
+ u8 dram_offset[3];
|
||||
+ u8 dram_size[3];
|
||||
+} __packed;
|
||||
+
|
||||
+enum aqr_fw_src {
|
||||
+ AQR_FW_SRC_NVMEM = 0,
|
||||
+ AQR_FW_SRC_FS,
|
||||
+};
|
||||
+
|
||||
+static const char * const aqr_fw_src_string[] = {
|
||||
+ [AQR_FW_SRC_NVMEM] = "NVMEM",
|
||||
+ [AQR_FW_SRC_FS] = "FS",
|
||||
+};
|
||||
+
|
||||
+/* AQR firmware doesn't have fixed offsets for iram and dram section
|
||||
+ * but instead provide an header with the offset to use on reading
|
||||
+ * and parsing the firmware.
|
||||
+ *
|
||||
+ * AQR firmware can't be trusted and each offset is validated to be
|
||||
+ * not negative and be in the size of the firmware itself.
|
||||
+ */
|
||||
+static bool aqr_fw_validate_get(size_t size, size_t offset, size_t get_size)
|
||||
+{
|
||||
+ return offset + get_size <= size;
|
||||
+}
|
||||
+
|
||||
+static int aqr_fw_get_be16(const u8 *data, size_t offset, size_t size, u16 *value)
|
||||
+{
|
||||
+ if (!aqr_fw_validate_get(size, offset, sizeof(u16)))
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ *value = get_unaligned_be16(data + offset);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int aqr_fw_get_le16(const u8 *data, size_t offset, size_t size, u16 *value)
|
||||
+{
|
||||
+ if (!aqr_fw_validate_get(size, offset, sizeof(u16)))
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ *value = get_unaligned_le16(data + offset);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int aqr_fw_get_le24(const u8 *data, size_t offset, size_t size, u32 *value)
|
||||
+{
|
||||
+ if (!aqr_fw_validate_get(size, offset, sizeof(u8) * 3))
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ *value = get_unaligned_le24(data + offset);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+/* load data into the phy's memory */
|
||||
+static int aqr_fw_load_memory(struct phy_device *phydev, u32 addr,
|
||||
+ const u8 *data, size_t len)
|
||||
+{
|
||||
+ u16 crc = 0, up_crc;
|
||||
+ size_t pos;
|
||||
+
|
||||
+ /* PHY expect addr in LE */
|
||||
+ addr = (__force u32)cpu_to_le32(addr);
|
||||
+
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE1,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE1_CRC_RESET);
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE3,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE3_MSW_ADDR(addr));
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE4,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE4_LSW_ADDR(addr));
|
||||
+
|
||||
+ /* We assume and enforce the size to be word aligned.
|
||||
+ * If a firmware that is not word aligned is found, please report upstream.
|
||||
+ */
|
||||
+ for (pos = 0; pos < len; pos += sizeof(u32)) {
|
||||
+ u32 word;
|
||||
+
|
||||
+ /* FW data is always stored in little-endian */
|
||||
+ word = get_unaligned((const u32 *)(data + pos));
|
||||
+
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_MAILBOX_INTERFACE5,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE5_MSW_DATA(word));
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_MAILBOX_INTERFACE6,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE6_LSW_DATA(word));
|
||||
+
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_MAILBOX_INTERFACE1,
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE1_EXECUTE |
|
||||
+ VEND1_GLOBAL_MAILBOX_INTERFACE1_WRITE);
|
||||
+
|
||||
+ /* calculate CRC as we load data to the mailbox.
|
||||
+ * We convert word to big-endian as PHY is BE and mailbox will
|
||||
+ * return a BE CRC.
|
||||
+ */
|
||||
+ word = (__force u32)cpu_to_be32(word);
|
||||
+ crc = crc_ccitt_false(crc, (u8 *)&word, sizeof(word));
|
||||
+ }
|
||||
+
|
||||
+ up_crc = phy_read_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_MAILBOX_INTERFACE2);
|
||||
+ if (crc != up_crc) {
|
||||
+ phydev_err(phydev, "CRC mismatch: calculated 0x%04x PHY 0x%04x\n",
|
||||
+ crc, up_crc);
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int aqr_fw_boot(struct phy_device *phydev, const u8 *data, size_t size,
|
||||
+ enum aqr_fw_src fw_src)
|
||||
+{
|
||||
+ u16 calculated_crc, read_crc, read_primary_offset;
|
||||
+ u32 iram_offset = 0, iram_size = 0;
|
||||
+ u32 dram_offset = 0, dram_size = 0;
|
||||
+ char version[VERSION_STRING_SIZE];
|
||||
+ u32 primary_offset = 0;
|
||||
+ int ret;
|
||||
+
|
||||
+ /* extract saved CRC at the end of the fw
|
||||
+ * CRC is saved in big-endian as PHY is BE
|
||||
+ */
|
||||
+ ret = aqr_fw_get_be16(data, size - sizeof(u16), size, &read_crc);
|
||||
+ if (ret) {
|
||||
+ phydev_err(phydev, "bad firmware CRC in firmware\n");
|
||||
+ return ret;
|
||||
+ }
|
||||
+ calculated_crc = crc_ccitt_false(0, data, size - sizeof(u16));
|
||||
+ if (read_crc != calculated_crc) {
|
||||
+ phydev_err(phydev, "bad firmware CRC: file 0x%04x calculated 0x%04x\n",
|
||||
+ read_crc, calculated_crc);
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ /* Get the primary offset to extract DRAM and IRAM sections. */
|
||||
+ ret = aqr_fw_get_le16(data, PRIMARY_OFFSET_OFFSET, size, &read_primary_offset);
|
||||
+ if (ret) {
|
||||
+ phydev_err(phydev, "bad primary offset in firmware\n");
|
||||
+ return ret;
|
||||
+ }
|
||||
+ primary_offset = PRIMARY_OFFSET(read_primary_offset);
|
||||
+
|
||||
+ /* Find the DRAM and IRAM sections within the firmware file.
|
||||
+ * Make sure the fw_header is correctly in the firmware.
|
||||
+ */
|
||||
+ if (!aqr_fw_validate_get(size, primary_offset + HEADER_OFFSET,
|
||||
+ sizeof(struct aqr_fw_header))) {
|
||||
+ phydev_err(phydev, "bad fw_header in firmware\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ /* offset are in LE and values needs to be converted to cpu endian */
|
||||
+ ret = aqr_fw_get_le24(data, primary_offset + HEADER_OFFSET +
|
||||
+ offsetof(struct aqr_fw_header, iram_offset),
|
||||
+ size, &iram_offset);
|
||||
+ if (ret) {
|
||||
+ phydev_err(phydev, "bad iram offset in firmware\n");
|
||||
+ return ret;
|
||||
+ }
|
||||
+ ret = aqr_fw_get_le24(data, primary_offset + HEADER_OFFSET +
|
||||
+ offsetof(struct aqr_fw_header, iram_size),
|
||||
+ size, &iram_size);
|
||||
+ if (ret) {
|
||||
+ phydev_err(phydev, "invalid iram size in firmware\n");
|
||||
+ return ret;
|
||||
+ }
|
||||
+ ret = aqr_fw_get_le24(data, primary_offset + HEADER_OFFSET +
|
||||
+ offsetof(struct aqr_fw_header, dram_offset),
|
||||
+ size, &dram_offset);
|
||||
+ if (ret) {
|
||||
+ phydev_err(phydev, "bad dram offset in firmware\n");
|
||||
+ return ret;
|
||||
+ }
|
||||
+ ret = aqr_fw_get_le24(data, primary_offset + HEADER_OFFSET +
|
||||
+ offsetof(struct aqr_fw_header, dram_size),
|
||||
+ size, &dram_size);
|
||||
+ if (ret) {
|
||||
+ phydev_err(phydev, "invalid dram size in firmware\n");
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ /* Increment the offset with the primary offset.
|
||||
+ * Validate iram/dram offset and size.
|
||||
+ */
|
||||
+ iram_offset += primary_offset;
|
||||
+ if (iram_size % sizeof(u32)) {
|
||||
+ phydev_err(phydev, "iram size if not aligned to word size. Please report this upstream!\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+ if (!aqr_fw_validate_get(size, iram_offset, iram_size)) {
|
||||
+ phydev_err(phydev, "invalid iram offset for iram size\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ dram_offset += primary_offset;
|
||||
+ if (dram_size % sizeof(u32)) {
|
||||
+ phydev_err(phydev, "dram size if not aligned to word size. Please report this upstream!\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+ if (!aqr_fw_validate_get(size, dram_offset, dram_size)) {
|
||||
+ phydev_err(phydev, "invalid iram offset for iram size\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ phydev_dbg(phydev, "primary %d IRAM offset=%d size=%d DRAM offset=%d size=%d\n",
|
||||
+ primary_offset, iram_offset, iram_size, dram_offset, dram_size);
|
||||
+
|
||||
+ if (!aqr_fw_validate_get(size, dram_offset + VERSION_STRING_OFFSET,
|
||||
+ VERSION_STRING_SIZE)) {
|
||||
+ phydev_err(phydev, "invalid version in firmware\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+ strscpy(version, (char *)data + dram_offset + VERSION_STRING_OFFSET,
|
||||
+ VERSION_STRING_SIZE);
|
||||
+ if (version[0] == '\0') {
|
||||
+ phydev_err(phydev, "invalid version in firmware\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+ phydev_info(phydev, "loading firmware version '%s' from '%s'\n", version,
|
||||
+ aqr_fw_src_string[fw_src]);
|
||||
+
|
||||
+ /* stall the microcprocessor */
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_CONTROL2,
|
||||
+ VEND1_GLOBAL_CONTROL2_UP_RUN_STALL | VEND1_GLOBAL_CONTROL2_UP_RUN_STALL_OVD);
|
||||
+
|
||||
+ phydev_dbg(phydev, "loading DRAM 0x%08x from offset=%d size=%d\n",
|
||||
+ DRAM_BASE_ADDR, dram_offset, dram_size);
|
||||
+ ret = aqr_fw_load_memory(phydev, DRAM_BASE_ADDR, data + dram_offset,
|
||||
+ dram_size);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ phydev_dbg(phydev, "loading IRAM 0x%08x from offset=%d size=%d\n",
|
||||
+ IRAM_BASE_ADDR, iram_offset, iram_size);
|
||||
+ ret = aqr_fw_load_memory(phydev, IRAM_BASE_ADDR, data + iram_offset,
|
||||
+ iram_size);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ /* make sure soft reset and low power mode are clear */
|
||||
+ phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_SC,
|
||||
+ VEND1_GLOBAL_SC_SOFT_RESET | VEND1_GLOBAL_SC_LOW_POWER);
|
||||
+
|
||||
+ /* Release the microprocessor. UP_RESET must be held for 100 usec. */
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_CONTROL2,
|
||||
+ VEND1_GLOBAL_CONTROL2_UP_RUN_STALL |
|
||||
+ VEND1_GLOBAL_CONTROL2_UP_RUN_STALL_OVD |
|
||||
+ VEND1_GLOBAL_CONTROL2_UP_RUN_STALL_RST);
|
||||
+ usleep_range(UP_RESET_SLEEP, UP_RESET_SLEEP * 2);
|
||||
+
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_CONTROL2,
|
||||
+ VEND1_GLOBAL_CONTROL2_UP_RUN_STALL_OVD);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int aqr_firmware_load_nvmem(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct nvmem_cell *cell;
|
||||
+ size_t size;
|
||||
+ u8 *buf;
|
||||
+ int ret;
|
||||
+
|
||||
+ cell = nvmem_cell_get(&phydev->mdio.dev, "firmware");
|
||||
+ if (IS_ERR(cell))
|
||||
+ return PTR_ERR(cell);
|
||||
+
|
||||
+ buf = nvmem_cell_read(cell, &size);
|
||||
+ if (IS_ERR(buf)) {
|
||||
+ ret = PTR_ERR(buf);
|
||||
+ goto exit;
|
||||
+ }
|
||||
+
|
||||
+ ret = aqr_fw_boot(phydev, buf, size, AQR_FW_SRC_NVMEM);
|
||||
+ if (ret)
|
||||
+ phydev_err(phydev, "firmware loading failed: %d\n", ret);
|
||||
+
|
||||
+ kfree(buf);
|
||||
+exit:
|
||||
+ nvmem_cell_put(cell);
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
+static int aqr_firmware_load_fs(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct device *dev = &phydev->mdio.dev;
|
||||
+ const struct firmware *fw;
|
||||
+ const char *fw_name;
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = of_property_read_string(dev->of_node, "firmware-name",
|
||||
+ &fw_name);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = request_firmware(&fw, fw_name, dev);
|
||||
+ if (ret) {
|
||||
+ phydev_err(phydev, "failed to find FW file %s (%d)\n",
|
||||
+ fw_name, ret);
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ ret = aqr_fw_boot(phydev, fw->data, fw->size, AQR_FW_SRC_FS);
|
||||
+ if (ret)
|
||||
+ phydev_err(phydev, "firmware loading failed: %d\n", ret);
|
||||
+
|
||||
+ release_firmware(fw);
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
+int aqr_firmware_load(struct phy_device *phydev)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ /* Check if the firmware is not already loaded by pooling
|
||||
+ * the current version returned by the PHY. If 0 is returned,
|
||||
+ * no firmware is loaded.
|
||||
+ */
|
||||
+ ret = phy_read_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_FW_ID);
|
||||
+ if (ret > 0)
|
||||
+ goto exit;
|
||||
+
|
||||
+ ret = aqr_firmware_load_nvmem(phydev);
|
||||
+ if (!ret)
|
||||
+ goto exit;
|
||||
+
|
||||
+ ret = aqr_firmware_load_fs(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+exit:
|
||||
+ return 0;
|
||||
+}
|
||||
--- a/drivers/net/phy/aquantia/aquantia_main.c
|
||||
+++ b/drivers/net/phy/aquantia/aquantia_main.c
|
||||
@@ -658,11 +658,17 @@ static int aqr107_resume(struct phy_devi
|
||||
|
||||
static int aqr107_probe(struct phy_device *phydev)
|
||||
{
|
||||
+ int ret;
|
||||
+
|
||||
phydev->priv = devm_kzalloc(&phydev->mdio.dev,
|
||||
sizeof(struct aqr107_priv), GFP_KERNEL);
|
||||
if (!phydev->priv)
|
||||
return -ENOMEM;
|
||||
|
||||
+ ret = aqr_firmware_load(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
return aqr_hwmon_probe(phydev);
|
||||
}
|
||||
|
@ -0,0 +1,69 @@
|
||||
From 6a3b8c573b5a152a6aa7a0b54c5e18b84c6ba6f5 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:49 +0100
|
||||
Subject: [PATCH 02/13] net: phy: at803x: move disable WOL to specific at8031
|
||||
probe
|
||||
|
||||
Move the WOL disable call to specific at8031 probe to make at803x_probe
|
||||
more generic and drop extra check for PHY ID.
|
||||
|
||||
Keep the same previous behaviour by first calling at803x_probe and then
|
||||
disabling WOL.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 27 +++++++++++++++++----------
|
||||
1 file changed, 17 insertions(+), 10 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -886,15 +886,6 @@ static int at803x_probe(struct phy_devic
|
||||
priv->is_fiber = true;
|
||||
break;
|
||||
}
|
||||
-
|
||||
- /* Disable WoL in 1588 register which is enabled
|
||||
- * by default
|
||||
- */
|
||||
- ret = phy_modify_mmd(phydev, MDIO_MMD_PCS,
|
||||
- AT803X_PHY_MMD3_WOL_CTRL,
|
||||
- AT803X_WOL_EN, 0);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
@@ -1591,6 +1582,22 @@ static int at803x_cable_test_start(struc
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int at8031_probe(struct phy_device *phydev)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = at803x_probe(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ /* Disable WoL in 1588 register which is enabled
|
||||
+ * by default
|
||||
+ */
|
||||
+ return phy_modify_mmd(phydev, MDIO_MMD_PCS,
|
||||
+ AT803X_PHY_MMD3_WOL_CTRL,
|
||||
+ AT803X_WOL_EN, 0);
|
||||
+}
|
||||
+
|
||||
static int qca83xx_config_init(struct phy_device *phydev)
|
||||
{
|
||||
u8 switch_revision;
|
||||
@@ -2092,7 +2099,7 @@ static struct phy_driver at803x_driver[]
|
||||
PHY_ID_MATCH_EXACT(ATH8031_PHY_ID),
|
||||
.name = "Qualcomm Atheros AR8031/AR8033",
|
||||
.flags = PHY_POLL_CABLE_TEST,
|
||||
- .probe = at803x_probe,
|
||||
+ .probe = at8031_probe,
|
||||
.config_init = at803x_config_init,
|
||||
.config_aneg = at803x_config_aneg,
|
||||
.soft_reset = genphy_soft_reset,
|
@ -0,0 +1,129 @@
|
||||
From 07b1ad83b9ed6db1735ba10baf67b7a565ac0cef Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:50 +0100
|
||||
Subject: [PATCH 03/13] net: phy: at803x: raname hw_stats functions to qca83xx
|
||||
specific name
|
||||
|
||||
The function and the struct related to hw_stats were specific to qca83xx
|
||||
PHY but were called following the convention in the driver of calling
|
||||
everything with at803x prefix.
|
||||
|
||||
To better organize the code, rename these function a more specific name
|
||||
to better describe that they are specific to 83xx PHY family.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 44 ++++++++++++++++++++--------------------
|
||||
1 file changed, 22 insertions(+), 22 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -295,7 +295,7 @@ struct at803x_hw_stat {
|
||||
enum stat_access_type access_type;
|
||||
};
|
||||
|
||||
-static struct at803x_hw_stat at803x_hw_stats[] = {
|
||||
+static struct at803x_hw_stat qca83xx_hw_stats[] = {
|
||||
{ "phy_idle_errors", 0xa, GENMASK(7, 0), PHY},
|
||||
{ "phy_receive_errors", 0x15, GENMASK(15, 0), PHY},
|
||||
{ "eee_wake_errors", 0x16, GENMASK(15, 0), MMD},
|
||||
@@ -311,7 +311,7 @@ struct at803x_priv {
|
||||
bool is_1000basex;
|
||||
struct regulator_dev *vddio_rdev;
|
||||
struct regulator_dev *vddh_rdev;
|
||||
- u64 stats[ARRAY_SIZE(at803x_hw_stats)];
|
||||
+ u64 stats[ARRAY_SIZE(qca83xx_hw_stats)];
|
||||
};
|
||||
|
||||
struct at803x_context {
|
||||
@@ -529,24 +529,24 @@ static void at803x_get_wol(struct phy_de
|
||||
wol->wolopts |= WAKE_MAGIC;
|
||||
}
|
||||
|
||||
-static int at803x_get_sset_count(struct phy_device *phydev)
|
||||
+static int qca83xx_get_sset_count(struct phy_device *phydev)
|
||||
{
|
||||
- return ARRAY_SIZE(at803x_hw_stats);
|
||||
+ return ARRAY_SIZE(qca83xx_hw_stats);
|
||||
}
|
||||
|
||||
-static void at803x_get_strings(struct phy_device *phydev, u8 *data)
|
||||
+static void qca83xx_get_strings(struct phy_device *phydev, u8 *data)
|
||||
{
|
||||
int i;
|
||||
|
||||
- for (i = 0; i < ARRAY_SIZE(at803x_hw_stats); i++) {
|
||||
+ for (i = 0; i < ARRAY_SIZE(qca83xx_hw_stats); i++) {
|
||||
strscpy(data + i * ETH_GSTRING_LEN,
|
||||
- at803x_hw_stats[i].string, ETH_GSTRING_LEN);
|
||||
+ qca83xx_hw_stats[i].string, ETH_GSTRING_LEN);
|
||||
}
|
||||
}
|
||||
|
||||
-static u64 at803x_get_stat(struct phy_device *phydev, int i)
|
||||
+static u64 qca83xx_get_stat(struct phy_device *phydev, int i)
|
||||
{
|
||||
- struct at803x_hw_stat stat = at803x_hw_stats[i];
|
||||
+ struct at803x_hw_stat stat = qca83xx_hw_stats[i];
|
||||
struct at803x_priv *priv = phydev->priv;
|
||||
int val;
|
||||
u64 ret;
|
||||
@@ -567,13 +567,13 @@ static u64 at803x_get_stat(struct phy_de
|
||||
return ret;
|
||||
}
|
||||
|
||||
-static void at803x_get_stats(struct phy_device *phydev,
|
||||
- struct ethtool_stats *stats, u64 *data)
|
||||
+static void qca83xx_get_stats(struct phy_device *phydev,
|
||||
+ struct ethtool_stats *stats, u64 *data)
|
||||
{
|
||||
int i;
|
||||
|
||||
- for (i = 0; i < ARRAY_SIZE(at803x_hw_stats); i++)
|
||||
- data[i] = at803x_get_stat(phydev, i);
|
||||
+ for (i = 0; i < ARRAY_SIZE(qca83xx_hw_stats); i++)
|
||||
+ data[i] = qca83xx_get_stat(phydev, i);
|
||||
}
|
||||
|
||||
static int at803x_suspend(struct phy_device *phydev)
|
||||
@@ -2175,9 +2175,9 @@ static struct phy_driver at803x_driver[]
|
||||
.flags = PHY_IS_INTERNAL,
|
||||
.config_init = qca83xx_config_init,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
- .get_sset_count = at803x_get_sset_count,
|
||||
- .get_strings = at803x_get_strings,
|
||||
- .get_stats = at803x_get_stats,
|
||||
+ .get_sset_count = qca83xx_get_sset_count,
|
||||
+ .get_strings = qca83xx_get_strings,
|
||||
+ .get_stats = qca83xx_get_stats,
|
||||
.suspend = qca83xx_suspend,
|
||||
.resume = qca83xx_resume,
|
||||
}, {
|
||||
@@ -2191,9 +2191,9 @@ static struct phy_driver at803x_driver[]
|
||||
.flags = PHY_IS_INTERNAL,
|
||||
.config_init = qca83xx_config_init,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
- .get_sset_count = at803x_get_sset_count,
|
||||
- .get_strings = at803x_get_strings,
|
||||
- .get_stats = at803x_get_stats,
|
||||
+ .get_sset_count = qca83xx_get_sset_count,
|
||||
+ .get_strings = qca83xx_get_strings,
|
||||
+ .get_stats = qca83xx_get_stats,
|
||||
.suspend = qca83xx_suspend,
|
||||
.resume = qca83xx_resume,
|
||||
}, {
|
||||
@@ -2207,9 +2207,9 @@ static struct phy_driver at803x_driver[]
|
||||
.flags = PHY_IS_INTERNAL,
|
||||
.config_init = qca83xx_config_init,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
- .get_sset_count = at803x_get_sset_count,
|
||||
- .get_strings = at803x_get_strings,
|
||||
- .get_stats = at803x_get_stats,
|
||||
+ .get_sset_count = qca83xx_get_sset_count,
|
||||
+ .get_strings = qca83xx_get_strings,
|
||||
+ .get_stats = qca83xx_get_stats,
|
||||
.suspend = qca83xx_suspend,
|
||||
.resume = qca83xx_resume,
|
||||
}, {
|
@ -0,0 +1,155 @@
|
||||
From d43cff3f82336c0bd965ea552232d9f4ddac71a6 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:51 +0100
|
||||
Subject: [PATCH 04/13] net: phy: at803x: move qca83xx specific check in
|
||||
dedicated functions
|
||||
|
||||
Rework qca83xx specific check to dedicated function to tidy things up
|
||||
and drop useless phy_id check.
|
||||
|
||||
Also drop an useless link_change_notify for QCA8337 as it did nothing an
|
||||
returned early.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 68 ++++++++++++++++++++++------------------
|
||||
1 file changed, 37 insertions(+), 31 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -1623,27 +1623,26 @@ static int qca83xx_config_init(struct ph
|
||||
break;
|
||||
}
|
||||
|
||||
+ /* Following original QCA sourcecode set port to prefer master */
|
||||
+ phy_set_bits(phydev, MII_CTRL1000, CTL1000_PREFER_MASTER);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca8327_config_init(struct phy_device *phydev)
|
||||
+{
|
||||
/* QCA8327 require DAC amplitude adjustment for 100m set to +6%.
|
||||
* Disable on init and enable only with 100m speed following
|
||||
* qca original source code.
|
||||
*/
|
||||
- if (phydev->drv->phy_id == QCA8327_A_PHY_ID ||
|
||||
- phydev->drv->phy_id == QCA8327_B_PHY_ID)
|
||||
- at803x_debug_reg_mask(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL,
|
||||
- QCA8327_DEBUG_MANU_CTRL_EN, 0);
|
||||
+ at803x_debug_reg_mask(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL,
|
||||
+ QCA8327_DEBUG_MANU_CTRL_EN, 0);
|
||||
|
||||
- /* Following original QCA sourcecode set port to prefer master */
|
||||
- phy_set_bits(phydev, MII_CTRL1000, CTL1000_PREFER_MASTER);
|
||||
-
|
||||
- return 0;
|
||||
+ return qca83xx_config_init(phydev);
|
||||
}
|
||||
|
||||
static void qca83xx_link_change_notify(struct phy_device *phydev)
|
||||
{
|
||||
- /* QCA8337 doesn't require DAC Amplitude adjustement */
|
||||
- if (phydev->drv->phy_id == QCA8337_PHY_ID)
|
||||
- return;
|
||||
-
|
||||
/* Set DAC Amplitude adjustment to +6% for 100m on link running */
|
||||
if (phydev->state == PHY_RUNNING) {
|
||||
if (phydev->speed == SPEED_100)
|
||||
@@ -1686,19 +1685,6 @@ static int qca83xx_resume(struct phy_dev
|
||||
|
||||
static int qca83xx_suspend(struct phy_device *phydev)
|
||||
{
|
||||
- u16 mask = 0;
|
||||
-
|
||||
- /* Only QCA8337 support actual suspend.
|
||||
- * QCA8327 cause port unreliability when phy suspend
|
||||
- * is set.
|
||||
- */
|
||||
- if (phydev->drv->phy_id == QCA8337_PHY_ID) {
|
||||
- genphy_suspend(phydev);
|
||||
- } else {
|
||||
- mask |= ~(BMCR_SPEED1000 | BMCR_FULLDPLX);
|
||||
- phy_modify(phydev, MII_BMCR, mask, 0);
|
||||
- }
|
||||
-
|
||||
at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_GREEN,
|
||||
AT803X_DEBUG_GATE_CLK_IN1000, 0);
|
||||
|
||||
@@ -1709,6 +1695,27 @@ static int qca83xx_suspend(struct phy_de
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int qca8337_suspend(struct phy_device *phydev)
|
||||
+{
|
||||
+ /* Only QCA8337 support actual suspend. */
|
||||
+ genphy_suspend(phydev);
|
||||
+
|
||||
+ return qca83xx_suspend(phydev);
|
||||
+}
|
||||
+
|
||||
+static int qca8327_suspend(struct phy_device *phydev)
|
||||
+{
|
||||
+ u16 mask = 0;
|
||||
+
|
||||
+ /* QCA8327 cause port unreliability when phy suspend
|
||||
+ * is set.
|
||||
+ */
|
||||
+ mask |= ~(BMCR_SPEED1000 | BMCR_FULLDPLX);
|
||||
+ phy_modify(phydev, MII_BMCR, mask, 0);
|
||||
+
|
||||
+ return qca83xx_suspend(phydev);
|
||||
+}
|
||||
+
|
||||
static int qca808x_phy_fast_retrain_config(struct phy_device *phydev)
|
||||
{
|
||||
int ret;
|
||||
@@ -2170,7 +2177,6 @@ static struct phy_driver at803x_driver[]
|
||||
.phy_id_mask = QCA8K_PHY_ID_MASK,
|
||||
.name = "Qualcomm Atheros 8337 internal PHY",
|
||||
/* PHY_GBIT_FEATURES */
|
||||
- .link_change_notify = qca83xx_link_change_notify,
|
||||
.probe = at803x_probe,
|
||||
.flags = PHY_IS_INTERNAL,
|
||||
.config_init = qca83xx_config_init,
|
||||
@@ -2178,7 +2184,7 @@ static struct phy_driver at803x_driver[]
|
||||
.get_sset_count = qca83xx_get_sset_count,
|
||||
.get_strings = qca83xx_get_strings,
|
||||
.get_stats = qca83xx_get_stats,
|
||||
- .suspend = qca83xx_suspend,
|
||||
+ .suspend = qca8337_suspend,
|
||||
.resume = qca83xx_resume,
|
||||
}, {
|
||||
/* QCA8327-A from switch QCA8327-AL1A */
|
||||
@@ -2189,12 +2195,12 @@ static struct phy_driver at803x_driver[]
|
||||
.link_change_notify = qca83xx_link_change_notify,
|
||||
.probe = at803x_probe,
|
||||
.flags = PHY_IS_INTERNAL,
|
||||
- .config_init = qca83xx_config_init,
|
||||
+ .config_init = qca8327_config_init,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
.get_sset_count = qca83xx_get_sset_count,
|
||||
.get_strings = qca83xx_get_strings,
|
||||
.get_stats = qca83xx_get_stats,
|
||||
- .suspend = qca83xx_suspend,
|
||||
+ .suspend = qca8327_suspend,
|
||||
.resume = qca83xx_resume,
|
||||
}, {
|
||||
/* QCA8327-B from switch QCA8327-BL1A */
|
||||
@@ -2205,12 +2211,12 @@ static struct phy_driver at803x_driver[]
|
||||
.link_change_notify = qca83xx_link_change_notify,
|
||||
.probe = at803x_probe,
|
||||
.flags = PHY_IS_INTERNAL,
|
||||
- .config_init = qca83xx_config_init,
|
||||
+ .config_init = qca8327_config_init,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
.get_sset_count = qca83xx_get_sset_count,
|
||||
.get_strings = qca83xx_get_strings,
|
||||
.get_stats = qca83xx_get_stats,
|
||||
- .suspend = qca83xx_suspend,
|
||||
+ .suspend = qca8327_suspend,
|
||||
.resume = qca83xx_resume,
|
||||
}, {
|
||||
/* Qualcomm QCA8081 */
|
@ -0,0 +1,94 @@
|
||||
From 900eef75cc5018e149c52fe305c9c3fe424c52a7 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:52 +0100
|
||||
Subject: [PATCH 05/13] net: phy: at803x: move specific DT option for at8031 to
|
||||
specific probe
|
||||
|
||||
Move specific DT options for at8031 to specific probe to tidy things up
|
||||
and make at803x_parse_dt more generic.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 55 ++++++++++++++++++++++------------------
|
||||
1 file changed, 31 insertions(+), 24 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -825,30 +825,6 @@ static int at803x_parse_dt(struct phy_de
|
||||
}
|
||||
}
|
||||
|
||||
- /* Only supported on AR8031/AR8033, the AR8030/AR8035 use strapping
|
||||
- * options.
|
||||
- */
|
||||
- if (phydev->drv->phy_id == ATH8031_PHY_ID) {
|
||||
- if (of_property_read_bool(node, "qca,keep-pll-enabled"))
|
||||
- priv->flags |= AT803X_KEEP_PLL_ENABLED;
|
||||
-
|
||||
- ret = at8031_register_regulators(phydev);
|
||||
- if (ret < 0)
|
||||
- return ret;
|
||||
-
|
||||
- ret = devm_regulator_get_enable_optional(&phydev->mdio.dev,
|
||||
- "vddio");
|
||||
- if (ret) {
|
||||
- phydev_err(phydev, "failed to get VDDIO regulator\n");
|
||||
- return ret;
|
||||
- }
|
||||
-
|
||||
- /* Only AR8031/8033 support 1000Base-X for SFP modules */
|
||||
- ret = phy_sfp_probe(phydev, &at803x_sfp_ops);
|
||||
- if (ret < 0)
|
||||
- return ret;
|
||||
- }
|
||||
-
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -1582,6 +1558,30 @@ static int at803x_cable_test_start(struc
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int at8031_parse_dt(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct device_node *node = phydev->mdio.dev.of_node;
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (of_property_read_bool(node, "qca,keep-pll-enabled"))
|
||||
+ priv->flags |= AT803X_KEEP_PLL_ENABLED;
|
||||
+
|
||||
+ ret = at8031_register_regulators(phydev);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = devm_regulator_get_enable_optional(&phydev->mdio.dev,
|
||||
+ "vddio");
|
||||
+ if (ret) {
|
||||
+ phydev_err(phydev, "failed to get VDDIO regulator\n");
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ /* Only AR8031/8033 support 1000Base-X for SFP modules */
|
||||
+ return phy_sfp_probe(phydev, &at803x_sfp_ops);
|
||||
+}
|
||||
+
|
||||
static int at8031_probe(struct phy_device *phydev)
|
||||
{
|
||||
int ret;
|
||||
@@ -1590,6 +1590,13 @@ static int at8031_probe(struct phy_devic
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
+ /* Only supported on AR8031/AR8033, the AR8030/AR8035 use strapping
|
||||
+ * options.
|
||||
+ */
|
||||
+ ret = at8031_parse_dt(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
/* Disable WoL in 1588 register which is enabled
|
||||
* by default
|
||||
*/
|
@ -0,0 +1,78 @@
|
||||
From 25d2ba94005fac18fe68878cddff59a67e115554 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:53 +0100
|
||||
Subject: [PATCH 06/13] net: phy: at803x: move specific at8031 probe mode check
|
||||
to dedicated probe
|
||||
|
||||
Move specific at8031 probe mode check to dedicated probe to make
|
||||
at803x_probe more generic and keep code tidy.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 39 +++++++++++++++++++--------------------
|
||||
1 file changed, 19 insertions(+), 20 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -844,26 +844,6 @@ static int at803x_probe(struct phy_devic
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
- if (phydev->drv->phy_id == ATH8031_PHY_ID) {
|
||||
- int ccr = phy_read(phydev, AT803X_REG_CHIP_CONFIG);
|
||||
- int mode_cfg;
|
||||
-
|
||||
- if (ccr < 0)
|
||||
- return ccr;
|
||||
- mode_cfg = ccr & AT803X_MODE_CFG_MASK;
|
||||
-
|
||||
- switch (mode_cfg) {
|
||||
- case AT803X_MODE_CFG_BX1000_RGMII_50OHM:
|
||||
- case AT803X_MODE_CFG_BX1000_RGMII_75OHM:
|
||||
- priv->is_1000basex = true;
|
||||
- fallthrough;
|
||||
- case AT803X_MODE_CFG_FX100_RGMII_50OHM:
|
||||
- case AT803X_MODE_CFG_FX100_RGMII_75OHM:
|
||||
- priv->is_fiber = true;
|
||||
- break;
|
||||
- }
|
||||
- }
|
||||
-
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -1584,6 +1564,9 @@ static int at8031_parse_dt(struct phy_de
|
||||
|
||||
static int at8031_probe(struct phy_device *phydev)
|
||||
{
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+ int mode_cfg;
|
||||
+ int ccr;
|
||||
int ret;
|
||||
|
||||
ret = at803x_probe(phydev);
|
||||
@@ -1597,6 +1580,22 @@ static int at8031_probe(struct phy_devic
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
+ ccr = phy_read(phydev, AT803X_REG_CHIP_CONFIG);
|
||||
+ if (ccr < 0)
|
||||
+ return ccr;
|
||||
+ mode_cfg = ccr & AT803X_MODE_CFG_MASK;
|
||||
+
|
||||
+ switch (mode_cfg) {
|
||||
+ case AT803X_MODE_CFG_BX1000_RGMII_50OHM:
|
||||
+ case AT803X_MODE_CFG_BX1000_RGMII_75OHM:
|
||||
+ priv->is_1000basex = true;
|
||||
+ fallthrough;
|
||||
+ case AT803X_MODE_CFG_FX100_RGMII_50OHM:
|
||||
+ case AT803X_MODE_CFG_FX100_RGMII_75OHM:
|
||||
+ priv->is_fiber = true;
|
||||
+ break;
|
||||
+ }
|
||||
+
|
||||
/* Disable WoL in 1588 register which is enabled
|
||||
* by default
|
||||
*/
|
@ -0,0 +1,86 @@
|
||||
From 3ae3bc426eaf57ca8f53d75777d9a5ef779bc7b7 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:54 +0100
|
||||
Subject: [PATCH 07/13] net: phy: at803x: move specific at8031 config_init to
|
||||
dedicated function
|
||||
|
||||
Move specific at8031 config_init to dedicated function to make
|
||||
at803x_config_init more generic and tidy things up.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 45 ++++++++++++++++++++++------------------
|
||||
1 file changed, 25 insertions(+), 20 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -951,27 +951,8 @@ static int at803x_hibernation_mode_confi
|
||||
|
||||
static int at803x_config_init(struct phy_device *phydev)
|
||||
{
|
||||
- struct at803x_priv *priv = phydev->priv;
|
||||
int ret;
|
||||
|
||||
- if (phydev->drv->phy_id == ATH8031_PHY_ID) {
|
||||
- /* Some bootloaders leave the fiber page selected.
|
||||
- * Switch to the appropriate page (fiber or copper), as otherwise we
|
||||
- * read the PHY capabilities from the wrong page.
|
||||
- */
|
||||
- phy_lock_mdio_bus(phydev);
|
||||
- ret = at803x_write_page(phydev,
|
||||
- priv->is_fiber ? AT803X_PAGE_FIBER :
|
||||
- AT803X_PAGE_COPPER);
|
||||
- phy_unlock_mdio_bus(phydev);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- ret = at8031_pll_config(phydev);
|
||||
- if (ret < 0)
|
||||
- return ret;
|
||||
- }
|
||||
-
|
||||
/* The RX and TX delay default is:
|
||||
* after HW reset: RX delay enabled and TX delay disabled
|
||||
* after SW reset: RX delay enabled, while TX delay retains the
|
||||
@@ -1604,6 +1585,30 @@ static int at8031_probe(struct phy_devic
|
||||
AT803X_WOL_EN, 0);
|
||||
}
|
||||
|
||||
+static int at8031_config_init(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+ int ret;
|
||||
+
|
||||
+ /* Some bootloaders leave the fiber page selected.
|
||||
+ * Switch to the appropriate page (fiber or copper), as otherwise we
|
||||
+ * read the PHY capabilities from the wrong page.
|
||||
+ */
|
||||
+ phy_lock_mdio_bus(phydev);
|
||||
+ ret = at803x_write_page(phydev,
|
||||
+ priv->is_fiber ? AT803X_PAGE_FIBER :
|
||||
+ AT803X_PAGE_COPPER);
|
||||
+ phy_unlock_mdio_bus(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = at8031_pll_config(phydev);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+
|
||||
+ return at803x_config_init(phydev);
|
||||
+}
|
||||
+
|
||||
static int qca83xx_config_init(struct phy_device *phydev)
|
||||
{
|
||||
u8 switch_revision;
|
||||
@@ -2113,7 +2118,7 @@ static struct phy_driver at803x_driver[]
|
||||
.name = "Qualcomm Atheros AR8031/AR8033",
|
||||
.flags = PHY_POLL_CABLE_TEST,
|
||||
.probe = at8031_probe,
|
||||
- .config_init = at803x_config_init,
|
||||
+ .config_init = at8031_config_init,
|
||||
.config_aneg = at803x_config_aneg,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
.set_wol = at803x_set_wol,
|
@ -0,0 +1,92 @@
|
||||
From 27b89c9dc1b0393090d68d651b82f30ad2696baa Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:55 +0100
|
||||
Subject: [PATCH 08/13] net: phy: at803x: move specific at8031 WOL bits to
|
||||
dedicated function
|
||||
|
||||
Move specific at8031 WOL enable/disable to dedicated function to make
|
||||
at803x_set_wol more generic.
|
||||
|
||||
This is needed in preparation for PHY driver split as qca8081 share the
|
||||
same function to toggle WOL settings.
|
||||
|
||||
In this new implementation WOL module in at8031 is enabled after the
|
||||
generic interrupt is setup. This should not cause any problem as the
|
||||
WOL_INT has a separate implementation and only relay on MAC bits.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 42 ++++++++++++++++++++++++----------------
|
||||
1 file changed, 25 insertions(+), 17 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -466,27 +466,11 @@ static int at803x_set_wol(struct phy_dev
|
||||
phy_write_mmd(phydev, MDIO_MMD_PCS, offsets[i],
|
||||
mac[(i * 2) + 1] | (mac[(i * 2)] << 8));
|
||||
|
||||
- /* Enable WOL function for 1588 */
|
||||
- if (phydev->drv->phy_id == ATH8031_PHY_ID) {
|
||||
- ret = phy_modify_mmd(phydev, MDIO_MMD_PCS,
|
||||
- AT803X_PHY_MMD3_WOL_CTRL,
|
||||
- 0, AT803X_WOL_EN);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
- }
|
||||
/* Enable WOL interrupt */
|
||||
ret = phy_modify(phydev, AT803X_INTR_ENABLE, 0, AT803X_INTR_ENABLE_WOL);
|
||||
if (ret)
|
||||
return ret;
|
||||
} else {
|
||||
- /* Disable WoL function for 1588 */
|
||||
- if (phydev->drv->phy_id == ATH8031_PHY_ID) {
|
||||
- ret = phy_modify_mmd(phydev, MDIO_MMD_PCS,
|
||||
- AT803X_PHY_MMD3_WOL_CTRL,
|
||||
- AT803X_WOL_EN, 0);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
- }
|
||||
/* Disable WOL interrupt */
|
||||
ret = phy_modify(phydev, AT803X_INTR_ENABLE, AT803X_INTR_ENABLE_WOL, 0);
|
||||
if (ret)
|
||||
@@ -1609,6 +1593,30 @@ static int at8031_config_init(struct phy
|
||||
return at803x_config_init(phydev);
|
||||
}
|
||||
|
||||
+static int at8031_set_wol(struct phy_device *phydev,
|
||||
+ struct ethtool_wolinfo *wol)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ /* First setup MAC address and enable WOL interrupt */
|
||||
+ ret = at803x_set_wol(phydev, wol);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ if (wol->wolopts & WAKE_MAGIC)
|
||||
+ /* Enable WOL function for 1588 */
|
||||
+ ret = phy_modify_mmd(phydev, MDIO_MMD_PCS,
|
||||
+ AT803X_PHY_MMD3_WOL_CTRL,
|
||||
+ 0, AT803X_WOL_EN);
|
||||
+ else
|
||||
+ /* Disable WoL function for 1588 */
|
||||
+ ret = phy_modify_mmd(phydev, MDIO_MMD_PCS,
|
||||
+ AT803X_PHY_MMD3_WOL_CTRL,
|
||||
+ AT803X_WOL_EN, 0);
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
static int qca83xx_config_init(struct phy_device *phydev)
|
||||
{
|
||||
u8 switch_revision;
|
||||
@@ -2121,7 +2129,7 @@ static struct phy_driver at803x_driver[]
|
||||
.config_init = at8031_config_init,
|
||||
.config_aneg = at803x_config_aneg,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
- .set_wol = at803x_set_wol,
|
||||
+ .set_wol = at8031_set_wol,
|
||||
.get_wol = at803x_get_wol,
|
||||
.suspend = at803x_suspend,
|
||||
.resume = at803x_resume,
|
@ -0,0 +1,78 @@
|
||||
From 30dd62191d3dd97c08f7f9dc9ce77ffab457e4fb Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:56 +0100
|
||||
Subject: [PATCH 09/13] net: phy: at803x: move specific at8031 config_intr to
|
||||
dedicated function
|
||||
|
||||
Move specific at8031 config_intr bits to dedicated function to make
|
||||
at803x_config_initr more generic.
|
||||
|
||||
This is needed in preparation for PHY driver split as qca8081 share the
|
||||
same function to setup interrupts.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 30 ++++++++++++++++++++++++------
|
||||
1 file changed, 24 insertions(+), 6 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -990,7 +990,6 @@ static int at803x_ack_interrupt(struct p
|
||||
|
||||
static int at803x_config_intr(struct phy_device *phydev)
|
||||
{
|
||||
- struct at803x_priv *priv = phydev->priv;
|
||||
int err;
|
||||
int value;
|
||||
|
||||
@@ -1007,10 +1006,6 @@ static int at803x_config_intr(struct phy
|
||||
value |= AT803X_INTR_ENABLE_DUPLEX_CHANGED;
|
||||
value |= AT803X_INTR_ENABLE_LINK_FAIL;
|
||||
value |= AT803X_INTR_ENABLE_LINK_SUCCESS;
|
||||
- if (priv->is_fiber) {
|
||||
- value |= AT803X_INTR_ENABLE_LINK_FAIL_BX;
|
||||
- value |= AT803X_INTR_ENABLE_LINK_SUCCESS_BX;
|
||||
- }
|
||||
|
||||
err = phy_write(phydev, AT803X_INTR_ENABLE, value);
|
||||
} else {
|
||||
@@ -1617,6 +1612,29 @@ static int at8031_set_wol(struct phy_dev
|
||||
return ret;
|
||||
}
|
||||
|
||||
+static int at8031_config_intr(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+ int err, value = 0;
|
||||
+
|
||||
+ if (phydev->interrupts == PHY_INTERRUPT_ENABLED &&
|
||||
+ priv->is_fiber) {
|
||||
+ /* Clear any pending interrupts */
|
||||
+ err = at803x_ack_interrupt(phydev);
|
||||
+ if (err)
|
||||
+ return err;
|
||||
+
|
||||
+ value |= AT803X_INTR_ENABLE_LINK_FAIL_BX;
|
||||
+ value |= AT803X_INTR_ENABLE_LINK_SUCCESS_BX;
|
||||
+
|
||||
+ err = phy_set_bits(phydev, AT803X_INTR_ENABLE, value);
|
||||
+ if (err)
|
||||
+ return err;
|
||||
+ }
|
||||
+
|
||||
+ return at803x_config_intr(phydev);
|
||||
+}
|
||||
+
|
||||
static int qca83xx_config_init(struct phy_device *phydev)
|
||||
{
|
||||
u8 switch_revision;
|
||||
@@ -2137,7 +2155,7 @@ static struct phy_driver at803x_driver[]
|
||||
.write_page = at803x_write_page,
|
||||
.get_features = at803x_get_features,
|
||||
.read_status = at803x_read_status,
|
||||
- .config_intr = at803x_config_intr,
|
||||
+ .config_intr = at8031_config_intr,
|
||||
.handle_interrupt = at803x_handle_interrupt,
|
||||
.get_tunable = at803x_get_tunable,
|
||||
.set_tunable = at803x_set_tunable,
|
@ -0,0 +1,78 @@
|
||||
From a5ab9d8e7ae0da8328ac1637a9755311508dc8ab Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:57 +0100
|
||||
Subject: [PATCH 10/13] net: phy: at803x: make at8031 related DT functions name
|
||||
more specific
|
||||
|
||||
Rename at8031 related DT function name to a more specific name
|
||||
referencing they are only related to at8031 and not to the generic
|
||||
at803x PHY family.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 16 ++++++++--------
|
||||
1 file changed, 8 insertions(+), 8 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -583,7 +583,7 @@ static int at803x_resume(struct phy_devi
|
||||
return phy_modify(phydev, MII_BMCR, BMCR_PDOWN | BMCR_ISOLATE, 0);
|
||||
}
|
||||
|
||||
-static int at803x_rgmii_reg_set_voltage_sel(struct regulator_dev *rdev,
|
||||
+static int at8031_rgmii_reg_set_voltage_sel(struct regulator_dev *rdev,
|
||||
unsigned int selector)
|
||||
{
|
||||
struct phy_device *phydev = rdev_get_drvdata(rdev);
|
||||
@@ -596,7 +596,7 @@ static int at803x_rgmii_reg_set_voltage_
|
||||
AT803X_DEBUG_RGMII_1V8, 0);
|
||||
}
|
||||
|
||||
-static int at803x_rgmii_reg_get_voltage_sel(struct regulator_dev *rdev)
|
||||
+static int at8031_rgmii_reg_get_voltage_sel(struct regulator_dev *rdev)
|
||||
{
|
||||
struct phy_device *phydev = rdev_get_drvdata(rdev);
|
||||
int val;
|
||||
@@ -610,8 +610,8 @@ static int at803x_rgmii_reg_get_voltage_
|
||||
|
||||
static const struct regulator_ops vddio_regulator_ops = {
|
||||
.list_voltage = regulator_list_voltage_table,
|
||||
- .set_voltage_sel = at803x_rgmii_reg_set_voltage_sel,
|
||||
- .get_voltage_sel = at803x_rgmii_reg_get_voltage_sel,
|
||||
+ .set_voltage_sel = at8031_rgmii_reg_set_voltage_sel,
|
||||
+ .get_voltage_sel = at8031_rgmii_reg_get_voltage_sel,
|
||||
};
|
||||
|
||||
static const unsigned int vddio_voltage_table[] = {
|
||||
@@ -666,7 +666,7 @@ static int at8031_register_regulators(st
|
||||
return 0;
|
||||
}
|
||||
|
||||
-static int at803x_sfp_insert(void *upstream, const struct sfp_eeprom_id *id)
|
||||
+static int at8031_sfp_insert(void *upstream, const struct sfp_eeprom_id *id)
|
||||
{
|
||||
struct phy_device *phydev = upstream;
|
||||
__ETHTOOL_DECLARE_LINK_MODE_MASK(phy_support);
|
||||
@@ -710,10 +710,10 @@ static int at803x_sfp_insert(void *upstr
|
||||
return 0;
|
||||
}
|
||||
|
||||
-static const struct sfp_upstream_ops at803x_sfp_ops = {
|
||||
+static const struct sfp_upstream_ops at8031_sfp_ops = {
|
||||
.attach = phy_sfp_attach,
|
||||
.detach = phy_sfp_detach,
|
||||
- .module_insert = at803x_sfp_insert,
|
||||
+ .module_insert = at8031_sfp_insert,
|
||||
};
|
||||
|
||||
static int at803x_parse_dt(struct phy_device *phydev)
|
||||
@@ -1519,7 +1519,7 @@ static int at8031_parse_dt(struct phy_de
|
||||
}
|
||||
|
||||
/* Only AR8031/8033 support 1000Base-X for SFP modules */
|
||||
- return phy_sfp_probe(phydev, &at803x_sfp_ops);
|
||||
+ return phy_sfp_probe(phydev, &at8031_sfp_ops);
|
||||
}
|
||||
|
||||
static int at8031_probe(struct phy_device *phydev)
|
@ -0,0 +1,297 @@
|
||||
From f932a6dc8bae0dae9645b5b1b4c65aed8a8acb2a Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:58 +0100
|
||||
Subject: [PATCH 11/13] net: phy: at803x: move at8031 functions in dedicated
|
||||
section
|
||||
|
||||
Move at8031 functions in dedicated section with dedicated at8031
|
||||
parse_dt and probe.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 266 +++++++++++++++++++--------------------
|
||||
1 file changed, 133 insertions(+), 133 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -583,139 +583,6 @@ static int at803x_resume(struct phy_devi
|
||||
return phy_modify(phydev, MII_BMCR, BMCR_PDOWN | BMCR_ISOLATE, 0);
|
||||
}
|
||||
|
||||
-static int at8031_rgmii_reg_set_voltage_sel(struct regulator_dev *rdev,
|
||||
- unsigned int selector)
|
||||
-{
|
||||
- struct phy_device *phydev = rdev_get_drvdata(rdev);
|
||||
-
|
||||
- if (selector)
|
||||
- return at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_1F,
|
||||
- 0, AT803X_DEBUG_RGMII_1V8);
|
||||
- else
|
||||
- return at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_1F,
|
||||
- AT803X_DEBUG_RGMII_1V8, 0);
|
||||
-}
|
||||
-
|
||||
-static int at8031_rgmii_reg_get_voltage_sel(struct regulator_dev *rdev)
|
||||
-{
|
||||
- struct phy_device *phydev = rdev_get_drvdata(rdev);
|
||||
- int val;
|
||||
-
|
||||
- val = at803x_debug_reg_read(phydev, AT803X_DEBUG_REG_1F);
|
||||
- if (val < 0)
|
||||
- return val;
|
||||
-
|
||||
- return (val & AT803X_DEBUG_RGMII_1V8) ? 1 : 0;
|
||||
-}
|
||||
-
|
||||
-static const struct regulator_ops vddio_regulator_ops = {
|
||||
- .list_voltage = regulator_list_voltage_table,
|
||||
- .set_voltage_sel = at8031_rgmii_reg_set_voltage_sel,
|
||||
- .get_voltage_sel = at8031_rgmii_reg_get_voltage_sel,
|
||||
-};
|
||||
-
|
||||
-static const unsigned int vddio_voltage_table[] = {
|
||||
- 1500000,
|
||||
- 1800000,
|
||||
-};
|
||||
-
|
||||
-static const struct regulator_desc vddio_desc = {
|
||||
- .name = "vddio",
|
||||
- .of_match = of_match_ptr("vddio-regulator"),
|
||||
- .n_voltages = ARRAY_SIZE(vddio_voltage_table),
|
||||
- .volt_table = vddio_voltage_table,
|
||||
- .ops = &vddio_regulator_ops,
|
||||
- .type = REGULATOR_VOLTAGE,
|
||||
- .owner = THIS_MODULE,
|
||||
-};
|
||||
-
|
||||
-static const struct regulator_ops vddh_regulator_ops = {
|
||||
-};
|
||||
-
|
||||
-static const struct regulator_desc vddh_desc = {
|
||||
- .name = "vddh",
|
||||
- .of_match = of_match_ptr("vddh-regulator"),
|
||||
- .n_voltages = 1,
|
||||
- .fixed_uV = 2500000,
|
||||
- .ops = &vddh_regulator_ops,
|
||||
- .type = REGULATOR_VOLTAGE,
|
||||
- .owner = THIS_MODULE,
|
||||
-};
|
||||
-
|
||||
-static int at8031_register_regulators(struct phy_device *phydev)
|
||||
-{
|
||||
- struct at803x_priv *priv = phydev->priv;
|
||||
- struct device *dev = &phydev->mdio.dev;
|
||||
- struct regulator_config config = { };
|
||||
-
|
||||
- config.dev = dev;
|
||||
- config.driver_data = phydev;
|
||||
-
|
||||
- priv->vddio_rdev = devm_regulator_register(dev, &vddio_desc, &config);
|
||||
- if (IS_ERR(priv->vddio_rdev)) {
|
||||
- phydev_err(phydev, "failed to register VDDIO regulator\n");
|
||||
- return PTR_ERR(priv->vddio_rdev);
|
||||
- }
|
||||
-
|
||||
- priv->vddh_rdev = devm_regulator_register(dev, &vddh_desc, &config);
|
||||
- if (IS_ERR(priv->vddh_rdev)) {
|
||||
- phydev_err(phydev, "failed to register VDDH regulator\n");
|
||||
- return PTR_ERR(priv->vddh_rdev);
|
||||
- }
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
-
|
||||
-static int at8031_sfp_insert(void *upstream, const struct sfp_eeprom_id *id)
|
||||
-{
|
||||
- struct phy_device *phydev = upstream;
|
||||
- __ETHTOOL_DECLARE_LINK_MODE_MASK(phy_support);
|
||||
- __ETHTOOL_DECLARE_LINK_MODE_MASK(sfp_support);
|
||||
- DECLARE_PHY_INTERFACE_MASK(interfaces);
|
||||
- phy_interface_t iface;
|
||||
-
|
||||
- linkmode_zero(phy_support);
|
||||
- phylink_set(phy_support, 1000baseX_Full);
|
||||
- phylink_set(phy_support, 1000baseT_Full);
|
||||
- phylink_set(phy_support, Autoneg);
|
||||
- phylink_set(phy_support, Pause);
|
||||
- phylink_set(phy_support, Asym_Pause);
|
||||
-
|
||||
- linkmode_zero(sfp_support);
|
||||
- sfp_parse_support(phydev->sfp_bus, id, sfp_support, interfaces);
|
||||
- /* Some modules support 10G modes as well as others we support.
|
||||
- * Mask out non-supported modes so the correct interface is picked.
|
||||
- */
|
||||
- linkmode_and(sfp_support, phy_support, sfp_support);
|
||||
-
|
||||
- if (linkmode_empty(sfp_support)) {
|
||||
- dev_err(&phydev->mdio.dev, "incompatible SFP module inserted\n");
|
||||
- return -EINVAL;
|
||||
- }
|
||||
-
|
||||
- iface = sfp_select_interface(phydev->sfp_bus, sfp_support);
|
||||
-
|
||||
- /* Only 1000Base-X is supported by AR8031/8033 as the downstream SerDes
|
||||
- * interface for use with SFP modules.
|
||||
- * However, some copper modules detected as having a preferred SGMII
|
||||
- * interface do default to and function in 1000Base-X mode, so just
|
||||
- * print a warning and allow such modules, as they may have some chance
|
||||
- * of working.
|
||||
- */
|
||||
- if (iface == PHY_INTERFACE_MODE_SGMII)
|
||||
- dev_warn(&phydev->mdio.dev, "module may not function if 1000Base-X not supported\n");
|
||||
- else if (iface != PHY_INTERFACE_MODE_1000BASEX)
|
||||
- return -EINVAL;
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
-
|
||||
-static const struct sfp_upstream_ops at8031_sfp_ops = {
|
||||
- .attach = phy_sfp_attach,
|
||||
- .detach = phy_sfp_detach,
|
||||
- .module_insert = at8031_sfp_insert,
|
||||
-};
|
||||
-
|
||||
static int at803x_parse_dt(struct phy_device *phydev)
|
||||
{
|
||||
struct device_node *node = phydev->mdio.dev.of_node;
|
||||
@@ -1498,6 +1365,139 @@ static int at803x_cable_test_start(struc
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int at8031_rgmii_reg_set_voltage_sel(struct regulator_dev *rdev,
|
||||
+ unsigned int selector)
|
||||
+{
|
||||
+ struct phy_device *phydev = rdev_get_drvdata(rdev);
|
||||
+
|
||||
+ if (selector)
|
||||
+ return at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_1F,
|
||||
+ 0, AT803X_DEBUG_RGMII_1V8);
|
||||
+ else
|
||||
+ return at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_1F,
|
||||
+ AT803X_DEBUG_RGMII_1V8, 0);
|
||||
+}
|
||||
+
|
||||
+static int at8031_rgmii_reg_get_voltage_sel(struct regulator_dev *rdev)
|
||||
+{
|
||||
+ struct phy_device *phydev = rdev_get_drvdata(rdev);
|
||||
+ int val;
|
||||
+
|
||||
+ val = at803x_debug_reg_read(phydev, AT803X_DEBUG_REG_1F);
|
||||
+ if (val < 0)
|
||||
+ return val;
|
||||
+
|
||||
+ return (val & AT803X_DEBUG_RGMII_1V8) ? 1 : 0;
|
||||
+}
|
||||
+
|
||||
+static const struct regulator_ops vddio_regulator_ops = {
|
||||
+ .list_voltage = regulator_list_voltage_table,
|
||||
+ .set_voltage_sel = at8031_rgmii_reg_set_voltage_sel,
|
||||
+ .get_voltage_sel = at8031_rgmii_reg_get_voltage_sel,
|
||||
+};
|
||||
+
|
||||
+static const unsigned int vddio_voltage_table[] = {
|
||||
+ 1500000,
|
||||
+ 1800000,
|
||||
+};
|
||||
+
|
||||
+static const struct regulator_desc vddio_desc = {
|
||||
+ .name = "vddio",
|
||||
+ .of_match = of_match_ptr("vddio-regulator"),
|
||||
+ .n_voltages = ARRAY_SIZE(vddio_voltage_table),
|
||||
+ .volt_table = vddio_voltage_table,
|
||||
+ .ops = &vddio_regulator_ops,
|
||||
+ .type = REGULATOR_VOLTAGE,
|
||||
+ .owner = THIS_MODULE,
|
||||
+};
|
||||
+
|
||||
+static const struct regulator_ops vddh_regulator_ops = {
|
||||
+};
|
||||
+
|
||||
+static const struct regulator_desc vddh_desc = {
|
||||
+ .name = "vddh",
|
||||
+ .of_match = of_match_ptr("vddh-regulator"),
|
||||
+ .n_voltages = 1,
|
||||
+ .fixed_uV = 2500000,
|
||||
+ .ops = &vddh_regulator_ops,
|
||||
+ .type = REGULATOR_VOLTAGE,
|
||||
+ .owner = THIS_MODULE,
|
||||
+};
|
||||
+
|
||||
+static int at8031_register_regulators(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+ struct device *dev = &phydev->mdio.dev;
|
||||
+ struct regulator_config config = { };
|
||||
+
|
||||
+ config.dev = dev;
|
||||
+ config.driver_data = phydev;
|
||||
+
|
||||
+ priv->vddio_rdev = devm_regulator_register(dev, &vddio_desc, &config);
|
||||
+ if (IS_ERR(priv->vddio_rdev)) {
|
||||
+ phydev_err(phydev, "failed to register VDDIO regulator\n");
|
||||
+ return PTR_ERR(priv->vddio_rdev);
|
||||
+ }
|
||||
+
|
||||
+ priv->vddh_rdev = devm_regulator_register(dev, &vddh_desc, &config);
|
||||
+ if (IS_ERR(priv->vddh_rdev)) {
|
||||
+ phydev_err(phydev, "failed to register VDDH regulator\n");
|
||||
+ return PTR_ERR(priv->vddh_rdev);
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int at8031_sfp_insert(void *upstream, const struct sfp_eeprom_id *id)
|
||||
+{
|
||||
+ struct phy_device *phydev = upstream;
|
||||
+ __ETHTOOL_DECLARE_LINK_MODE_MASK(phy_support);
|
||||
+ __ETHTOOL_DECLARE_LINK_MODE_MASK(sfp_support);
|
||||
+ DECLARE_PHY_INTERFACE_MASK(interfaces);
|
||||
+ phy_interface_t iface;
|
||||
+
|
||||
+ linkmode_zero(phy_support);
|
||||
+ phylink_set(phy_support, 1000baseX_Full);
|
||||
+ phylink_set(phy_support, 1000baseT_Full);
|
||||
+ phylink_set(phy_support, Autoneg);
|
||||
+ phylink_set(phy_support, Pause);
|
||||
+ phylink_set(phy_support, Asym_Pause);
|
||||
+
|
||||
+ linkmode_zero(sfp_support);
|
||||
+ sfp_parse_support(phydev->sfp_bus, id, sfp_support, interfaces);
|
||||
+ /* Some modules support 10G modes as well as others we support.
|
||||
+ * Mask out non-supported modes so the correct interface is picked.
|
||||
+ */
|
||||
+ linkmode_and(sfp_support, phy_support, sfp_support);
|
||||
+
|
||||
+ if (linkmode_empty(sfp_support)) {
|
||||
+ dev_err(&phydev->mdio.dev, "incompatible SFP module inserted\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ iface = sfp_select_interface(phydev->sfp_bus, sfp_support);
|
||||
+
|
||||
+ /* Only 1000Base-X is supported by AR8031/8033 as the downstream SerDes
|
||||
+ * interface for use with SFP modules.
|
||||
+ * However, some copper modules detected as having a preferred SGMII
|
||||
+ * interface do default to and function in 1000Base-X mode, so just
|
||||
+ * print a warning and allow such modules, as they may have some chance
|
||||
+ * of working.
|
||||
+ */
|
||||
+ if (iface == PHY_INTERFACE_MODE_SGMII)
|
||||
+ dev_warn(&phydev->mdio.dev, "module may not function if 1000Base-X not supported\n");
|
||||
+ else if (iface != PHY_INTERFACE_MODE_1000BASEX)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static const struct sfp_upstream_ops at8031_sfp_ops = {
|
||||
+ .attach = phy_sfp_attach,
|
||||
+ .detach = phy_sfp_detach,
|
||||
+ .module_insert = at8031_sfp_insert,
|
||||
+};
|
||||
+
|
||||
static int at8031_parse_dt(struct phy_device *phydev)
|
||||
{
|
||||
struct device_node *node = phydev->mdio.dev.of_node;
|
@ -0,0 +1,114 @@
|
||||
From 21a2802a8365cfa82cc02187c1f95136d85592ad Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:51:59 +0100
|
||||
Subject: [PATCH 12/13] net: phy: at803x: move at8035 specific DT parse to
|
||||
dedicated probe
|
||||
|
||||
Move at8035 specific DT parse for clock out frequency to dedicated probe
|
||||
to make at803x probe function more generic.
|
||||
|
||||
This is to tidy code and no behaviour change are intended.
|
||||
|
||||
Detection logic is changed, we check if the clk 25m mask is set and if
|
||||
it's not zero, we assume the qca,clk-out-frequency property is set.
|
||||
|
||||
The property is checked in the generic at803x_parse_dt called by
|
||||
at803x_probe.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 60 +++++++++++++++++++++++++++-------------
|
||||
1 file changed, 41 insertions(+), 19 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -638,23 +638,6 @@ static int at803x_parse_dt(struct phy_de
|
||||
|
||||
priv->clk_25m_reg |= FIELD_PREP(AT803X_CLK_OUT_MASK, sel);
|
||||
priv->clk_25m_mask |= AT803X_CLK_OUT_MASK;
|
||||
-
|
||||
- /* Fixup for the AR8030/AR8035. This chip has another mask and
|
||||
- * doesn't support the DSP reference. Eg. the lowest bit of the
|
||||
- * mask. The upper two bits select the same frequencies. Mask
|
||||
- * the lowest bit here.
|
||||
- *
|
||||
- * Warning:
|
||||
- * There was no datasheet for the AR8030 available so this is
|
||||
- * just a guess. But the AR8035 is listed as pin compatible
|
||||
- * to the AR8030 so there might be a good chance it works on
|
||||
- * the AR8030 too.
|
||||
- */
|
||||
- if (phydev->drv->phy_id == ATH8030_PHY_ID ||
|
||||
- phydev->drv->phy_id == ATH8035_PHY_ID) {
|
||||
- priv->clk_25m_reg &= AT8035_CLK_OUT_MASK;
|
||||
- priv->clk_25m_mask &= AT8035_CLK_OUT_MASK;
|
||||
- }
|
||||
}
|
||||
|
||||
ret = of_property_read_u32(node, "qca,clk-out-strength", &strength);
|
||||
@@ -1635,6 +1618,45 @@ static int at8031_config_intr(struct phy
|
||||
return at803x_config_intr(phydev);
|
||||
}
|
||||
|
||||
+static int at8035_parse_dt(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+
|
||||
+ /* Mask is set by the generic at803x_parse_dt
|
||||
+ * if property is set. Assume property is set
|
||||
+ * with the mask not zero.
|
||||
+ */
|
||||
+ if (priv->clk_25m_mask) {
|
||||
+ /* Fixup for the AR8030/AR8035. This chip has another mask and
|
||||
+ * doesn't support the DSP reference. Eg. the lowest bit of the
|
||||
+ * mask. The upper two bits select the same frequencies. Mask
|
||||
+ * the lowest bit here.
|
||||
+ *
|
||||
+ * Warning:
|
||||
+ * There was no datasheet for the AR8030 available so this is
|
||||
+ * just a guess. But the AR8035 is listed as pin compatible
|
||||
+ * to the AR8030 so there might be a good chance it works on
|
||||
+ * the AR8030 too.
|
||||
+ */
|
||||
+ priv->clk_25m_reg &= AT8035_CLK_OUT_MASK;
|
||||
+ priv->clk_25m_mask &= AT8035_CLK_OUT_MASK;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+/* AR8030 and AR8035 shared the same special mask for clk_25m */
|
||||
+static int at8035_probe(struct phy_device *phydev)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = at803x_probe(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ return at8035_parse_dt(phydev);
|
||||
+}
|
||||
+
|
||||
static int qca83xx_config_init(struct phy_device *phydev)
|
||||
{
|
||||
u8 switch_revision;
|
||||
@@ -2107,7 +2129,7 @@ static struct phy_driver at803x_driver[]
|
||||
PHY_ID_MATCH_EXACT(ATH8035_PHY_ID),
|
||||
.name = "Qualcomm Atheros AR8035",
|
||||
.flags = PHY_POLL_CABLE_TEST,
|
||||
- .probe = at803x_probe,
|
||||
+ .probe = at8035_probe,
|
||||
.config_aneg = at803x_config_aneg,
|
||||
.config_init = at803x_config_init,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
@@ -2128,7 +2150,7 @@ static struct phy_driver at803x_driver[]
|
||||
.phy_id = ATH8030_PHY_ID,
|
||||
.name = "Qualcomm Atheros AR8030",
|
||||
.phy_id_mask = AT8030_PHY_ID_MASK,
|
||||
- .probe = at803x_probe,
|
||||
+ .probe = at8035_probe,
|
||||
.config_init = at803x_config_init,
|
||||
.link_change_notify = at803x_link_change_notify,
|
||||
.set_wol = at803x_set_wol,
|
@ -0,0 +1,219 @@
|
||||
From ef9df47b449e32e06501a11272809be49019bdb6 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 8 Dec 2023 15:52:00 +0100
|
||||
Subject: [PATCH 13/13] net: phy: at803x: drop specific PHY ID check from cable
|
||||
test functions
|
||||
|
||||
Drop specific PHY ID check for cable test functions for at803x. This is
|
||||
done to make functions more generic. While at it better describe what
|
||||
the functions does by using more symbolic function names.
|
||||
|
||||
PHYs that requires to set additional reg are moved to specific function
|
||||
calling the more generic one.
|
||||
|
||||
cdt_start and cdt_wait_for_completion are changed to take an additional
|
||||
arg to pass specific values specific to the PHY.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 95 +++++++++++++++++++++-------------------
|
||||
1 file changed, 50 insertions(+), 45 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -1222,31 +1222,16 @@ static int at803x_cdt_fault_length(u16 s
|
||||
return (dt * 824) / 10;
|
||||
}
|
||||
|
||||
-static int at803x_cdt_start(struct phy_device *phydev, int pair)
|
||||
+static int at803x_cdt_start(struct phy_device *phydev,
|
||||
+ u32 cdt_start)
|
||||
{
|
||||
- u16 cdt;
|
||||
-
|
||||
- /* qca8081 takes the different bit 15 to enable CDT test */
|
||||
- if (phydev->drv->phy_id == QCA8081_PHY_ID)
|
||||
- cdt = QCA808X_CDT_ENABLE_TEST |
|
||||
- QCA808X_CDT_LENGTH_UNIT |
|
||||
- QCA808X_CDT_INTER_CHECK_DIS;
|
||||
- else
|
||||
- cdt = FIELD_PREP(AT803X_CDT_MDI_PAIR_MASK, pair) |
|
||||
- AT803X_CDT_ENABLE_TEST;
|
||||
-
|
||||
- return phy_write(phydev, AT803X_CDT, cdt);
|
||||
+ return phy_write(phydev, AT803X_CDT, cdt_start);
|
||||
}
|
||||
|
||||
-static int at803x_cdt_wait_for_completion(struct phy_device *phydev)
|
||||
+static int at803x_cdt_wait_for_completion(struct phy_device *phydev,
|
||||
+ u32 cdt_en)
|
||||
{
|
||||
int val, ret;
|
||||
- u16 cdt_en;
|
||||
-
|
||||
- if (phydev->drv->phy_id == QCA8081_PHY_ID)
|
||||
- cdt_en = QCA808X_CDT_ENABLE_TEST;
|
||||
- else
|
||||
- cdt_en = AT803X_CDT_ENABLE_TEST;
|
||||
|
||||
/* One test run takes about 25ms */
|
||||
ret = phy_read_poll_timeout(phydev, AT803X_CDT, val,
|
||||
@@ -1266,11 +1251,13 @@ static int at803x_cable_test_one_pair(st
|
||||
};
|
||||
int ret, val;
|
||||
|
||||
- ret = at803x_cdt_start(phydev, pair);
|
||||
+ val = FIELD_PREP(AT803X_CDT_MDI_PAIR_MASK, pair) |
|
||||
+ AT803X_CDT_ENABLE_TEST;
|
||||
+ ret = at803x_cdt_start(phydev, val);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
- ret = at803x_cdt_wait_for_completion(phydev);
|
||||
+ ret = at803x_cdt_wait_for_completion(phydev, AT803X_CDT_ENABLE_TEST);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@@ -1292,19 +1279,11 @@ static int at803x_cable_test_one_pair(st
|
||||
}
|
||||
|
||||
static int at803x_cable_test_get_status(struct phy_device *phydev,
|
||||
- bool *finished)
|
||||
+ bool *finished, unsigned long pair_mask)
|
||||
{
|
||||
- unsigned long pair_mask;
|
||||
int retries = 20;
|
||||
int pair, ret;
|
||||
|
||||
- if (phydev->phy_id == ATH9331_PHY_ID ||
|
||||
- phydev->phy_id == ATH8032_PHY_ID ||
|
||||
- phydev->phy_id == QCA9561_PHY_ID)
|
||||
- pair_mask = 0x3;
|
||||
- else
|
||||
- pair_mask = 0xf;
|
||||
-
|
||||
*finished = false;
|
||||
|
||||
/* According to the datasheet the CDT can be performed when
|
||||
@@ -1331,7 +1310,7 @@ static int at803x_cable_test_get_status(
|
||||
return 0;
|
||||
}
|
||||
|
||||
-static int at803x_cable_test_start(struct phy_device *phydev)
|
||||
+static void at803x_cable_test_autoneg(struct phy_device *phydev)
|
||||
{
|
||||
/* Enable auto-negotiation, but advertise no capabilities, no link
|
||||
* will be established. A restart of the auto-negotiation is not
|
||||
@@ -1339,11 +1318,11 @@ static int at803x_cable_test_start(struc
|
||||
*/
|
||||
phy_write(phydev, MII_BMCR, BMCR_ANENABLE);
|
||||
phy_write(phydev, MII_ADVERTISE, ADVERTISE_CSMA);
|
||||
- if (phydev->phy_id != ATH9331_PHY_ID &&
|
||||
- phydev->phy_id != ATH8032_PHY_ID &&
|
||||
- phydev->phy_id != QCA9561_PHY_ID)
|
||||
- phy_write(phydev, MII_CTRL1000, 0);
|
||||
+}
|
||||
|
||||
+static int at803x_cable_test_start(struct phy_device *phydev)
|
||||
+{
|
||||
+ at803x_cable_test_autoneg(phydev);
|
||||
/* we do all the (time consuming) work later */
|
||||
return 0;
|
||||
}
|
||||
@@ -1618,6 +1597,29 @@ static int at8031_config_intr(struct phy
|
||||
return at803x_config_intr(phydev);
|
||||
}
|
||||
|
||||
+/* AR8031 and AR8035 share the same cable test get status reg */
|
||||
+static int at8031_cable_test_get_status(struct phy_device *phydev,
|
||||
+ bool *finished)
|
||||
+{
|
||||
+ return at803x_cable_test_get_status(phydev, finished, 0xf);
|
||||
+}
|
||||
+
|
||||
+/* AR8031 and AR8035 share the same cable test start logic */
|
||||
+static int at8031_cable_test_start(struct phy_device *phydev)
|
||||
+{
|
||||
+ at803x_cable_test_autoneg(phydev);
|
||||
+ phy_write(phydev, MII_CTRL1000, 0);
|
||||
+ /* we do all the (time consuming) work later */
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+/* AR8032, AR9331 and QCA9561 share the same cable test get status reg */
|
||||
+static int at8032_cable_test_get_status(struct phy_device *phydev,
|
||||
+ bool *finished)
|
||||
+{
|
||||
+ return at803x_cable_test_get_status(phydev, finished, 0x3);
|
||||
+}
|
||||
+
|
||||
static int at8035_parse_dt(struct phy_device *phydev)
|
||||
{
|
||||
struct at803x_priv *priv = phydev->priv;
|
||||
@@ -2041,11 +2043,14 @@ static int qca808x_cable_test_get_status
|
||||
|
||||
*finished = false;
|
||||
|
||||
- ret = at803x_cdt_start(phydev, 0);
|
||||
+ val = QCA808X_CDT_ENABLE_TEST |
|
||||
+ QCA808X_CDT_LENGTH_UNIT |
|
||||
+ QCA808X_CDT_INTER_CHECK_DIS;
|
||||
+ ret = at803x_cdt_start(phydev, val);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
- ret = at803x_cdt_wait_for_completion(phydev);
|
||||
+ ret = at803x_cdt_wait_for_completion(phydev, QCA808X_CDT_ENABLE_TEST);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@@ -2143,8 +2148,8 @@ static struct phy_driver at803x_driver[]
|
||||
.handle_interrupt = at803x_handle_interrupt,
|
||||
.get_tunable = at803x_get_tunable,
|
||||
.set_tunable = at803x_set_tunable,
|
||||
- .cable_test_start = at803x_cable_test_start,
|
||||
- .cable_test_get_status = at803x_cable_test_get_status,
|
||||
+ .cable_test_start = at8031_cable_test_start,
|
||||
+ .cable_test_get_status = at8031_cable_test_get_status,
|
||||
}, {
|
||||
/* Qualcomm Atheros AR8030 */
|
||||
.phy_id = ATH8030_PHY_ID,
|
||||
@@ -2181,8 +2186,8 @@ static struct phy_driver at803x_driver[]
|
||||
.handle_interrupt = at803x_handle_interrupt,
|
||||
.get_tunable = at803x_get_tunable,
|
||||
.set_tunable = at803x_set_tunable,
|
||||
- .cable_test_start = at803x_cable_test_start,
|
||||
- .cable_test_get_status = at803x_cable_test_get_status,
|
||||
+ .cable_test_start = at8031_cable_test_start,
|
||||
+ .cable_test_get_status = at8031_cable_test_get_status,
|
||||
}, {
|
||||
/* Qualcomm Atheros AR8032 */
|
||||
PHY_ID_MATCH_EXACT(ATH8032_PHY_ID),
|
||||
@@ -2197,7 +2202,7 @@ static struct phy_driver at803x_driver[]
|
||||
.config_intr = at803x_config_intr,
|
||||
.handle_interrupt = at803x_handle_interrupt,
|
||||
.cable_test_start = at803x_cable_test_start,
|
||||
- .cable_test_get_status = at803x_cable_test_get_status,
|
||||
+ .cable_test_get_status = at8032_cable_test_get_status,
|
||||
}, {
|
||||
/* ATHEROS AR9331 */
|
||||
PHY_ID_MATCH_EXACT(ATH9331_PHY_ID),
|
||||
@@ -2210,7 +2215,7 @@ static struct phy_driver at803x_driver[]
|
||||
.config_intr = at803x_config_intr,
|
||||
.handle_interrupt = at803x_handle_interrupt,
|
||||
.cable_test_start = at803x_cable_test_start,
|
||||
- .cable_test_get_status = at803x_cable_test_get_status,
|
||||
+ .cable_test_get_status = at8032_cable_test_get_status,
|
||||
.read_status = at803x_read_status,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
.config_aneg = at803x_config_aneg,
|
||||
@@ -2226,7 +2231,7 @@ static struct phy_driver at803x_driver[]
|
||||
.config_intr = at803x_config_intr,
|
||||
.handle_interrupt = at803x_handle_interrupt,
|
||||
.cable_test_start = at803x_cable_test_start,
|
||||
- .cable_test_get_status = at803x_cable_test_get_status,
|
||||
+ .cable_test_get_status = at8032_cable_test_get_status,
|
||||
.read_status = at803x_read_status,
|
||||
.soft_reset = genphy_soft_reset,
|
||||
.config_aneg = at803x_config_aneg,
|
@ -0,0 +1,116 @@
|
||||
From 8e732f1c6f2dc5e18f766d0f1b11df9db2dd044a Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 14 Dec 2023 01:44:31 +0100
|
||||
Subject: [PATCH 1/2] net: phy: at803x: move specific qca808x config_aneg to
|
||||
dedicated function
|
||||
|
||||
Move specific qca808x config_aneg to dedicated function to permit easier
|
||||
split of qca808x portion from at803x driver.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 66 ++++++++++++++++++++++++----------------
|
||||
1 file changed, 40 insertions(+), 26 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -1045,9 +1045,8 @@ static int at803x_config_mdix(struct phy
|
||||
FIELD_PREP(AT803X_SFC_MDI_CROSSOVER_MODE_M, val));
|
||||
}
|
||||
|
||||
-static int at803x_config_aneg(struct phy_device *phydev)
|
||||
+static int at803x_prepare_config_aneg(struct phy_device *phydev)
|
||||
{
|
||||
- struct at803x_priv *priv = phydev->priv;
|
||||
int ret;
|
||||
|
||||
ret = at803x_config_mdix(phydev, phydev->mdix_ctrl);
|
||||
@@ -1064,33 +1063,22 @@ static int at803x_config_aneg(struct phy
|
||||
return ret;
|
||||
}
|
||||
|
||||
- if (priv->is_1000basex)
|
||||
- return genphy_c37_config_aneg(phydev);
|
||||
-
|
||||
- /* Do not restart auto-negotiation by setting ret to 0 defautly,
|
||||
- * when calling __genphy_config_aneg later.
|
||||
- */
|
||||
- ret = 0;
|
||||
-
|
||||
- if (phydev->drv->phy_id == QCA8081_PHY_ID) {
|
||||
- int phy_ctrl = 0;
|
||||
+ return 0;
|
||||
+}
|
||||
|
||||
- /* The reg MII_BMCR also needs to be configured for force mode, the
|
||||
- * genphy_config_aneg is also needed.
|
||||
- */
|
||||
- if (phydev->autoneg == AUTONEG_DISABLE)
|
||||
- genphy_c45_pma_setup_forced(phydev);
|
||||
+static int at803x_config_aneg(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+ int ret;
|
||||
|
||||
- if (linkmode_test_bit(ETHTOOL_LINK_MODE_2500baseT_Full_BIT, phydev->advertising))
|
||||
- phy_ctrl = MDIO_AN_10GBT_CTRL_ADV2_5G;
|
||||
+ ret = at803x_prepare_config_aneg(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
|
||||
- ret = phy_modify_mmd_changed(phydev, MDIO_MMD_AN, MDIO_AN_10GBT_CTRL,
|
||||
- MDIO_AN_10GBT_CTRL_ADV2_5G, phy_ctrl);
|
||||
- if (ret < 0)
|
||||
- return ret;
|
||||
- }
|
||||
+ if (priv->is_1000basex)
|
||||
+ return genphy_c37_config_aneg(phydev);
|
||||
|
||||
- return __genphy_config_aneg(phydev, ret);
|
||||
+ return genphy_config_aneg(phydev);
|
||||
}
|
||||
|
||||
static int at803x_get_downshift(struct phy_device *phydev, u8 *d)
|
||||
@@ -2118,6 +2106,32 @@ static int qca808x_get_features(struct p
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int qca808x_config_aneg(struct phy_device *phydev)
|
||||
+{
|
||||
+ int phy_ctrl = 0;
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = at803x_prepare_config_aneg(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ /* The reg MII_BMCR also needs to be configured for force mode, the
|
||||
+ * genphy_config_aneg is also needed.
|
||||
+ */
|
||||
+ if (phydev->autoneg == AUTONEG_DISABLE)
|
||||
+ genphy_c45_pma_setup_forced(phydev);
|
||||
+
|
||||
+ if (linkmode_test_bit(ETHTOOL_LINK_MODE_2500baseT_Full_BIT, phydev->advertising))
|
||||
+ phy_ctrl = MDIO_AN_10GBT_CTRL_ADV2_5G;
|
||||
+
|
||||
+ ret = phy_modify_mmd_changed(phydev, MDIO_MMD_AN, MDIO_AN_10GBT_CTRL,
|
||||
+ MDIO_AN_10GBT_CTRL_ADV2_5G, phy_ctrl);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+
|
||||
+ return __genphy_config_aneg(phydev, ret);
|
||||
+}
|
||||
+
|
||||
static void qca808x_link_change_notify(struct phy_device *phydev)
|
||||
{
|
||||
/* Assert interface sgmii fifo on link down, deassert it on link up,
|
||||
@@ -2295,7 +2309,7 @@ static struct phy_driver at803x_driver[]
|
||||
.set_wol = at803x_set_wol,
|
||||
.get_wol = at803x_get_wol,
|
||||
.get_features = qca808x_get_features,
|
||||
- .config_aneg = at803x_config_aneg,
|
||||
+ .config_aneg = qca808x_config_aneg,
|
||||
.suspend = genphy_suspend,
|
||||
.resume = genphy_resume,
|
||||
.read_status = qca808x_read_status,
|
@ -0,0 +1,97 @@
|
||||
From 38eb804e8458ba181a03a0498ce4bf84eebd1931 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 14 Dec 2023 01:44:32 +0100
|
||||
Subject: [PATCH 2/2] net: phy: at803x: make read specific status function more
|
||||
generic
|
||||
|
||||
Rework read specific status function to be more generic. The function
|
||||
apply different speed mask based on the PHY ID. Make it more generic by
|
||||
adding an additional arg to pass the specific speed (ss) mask and use
|
||||
the provided mask to parse the speed value.
|
||||
|
||||
This is needed to permit an easier deatch of qca808x code from the
|
||||
at803x driver.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 26 ++++++++++++++++++--------
|
||||
1 file changed, 18 insertions(+), 8 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -301,6 +301,11 @@ static struct at803x_hw_stat qca83xx_hw_
|
||||
{ "eee_wake_errors", 0x16, GENMASK(15, 0), MMD},
|
||||
};
|
||||
|
||||
+struct at803x_ss_mask {
|
||||
+ u16 speed_mask;
|
||||
+ u8 speed_shift;
|
||||
+};
|
||||
+
|
||||
struct at803x_priv {
|
||||
int flags;
|
||||
u16 clk_25m_reg;
|
||||
@@ -921,7 +926,8 @@ static void at803x_link_change_notify(st
|
||||
}
|
||||
}
|
||||
|
||||
-static int at803x_read_specific_status(struct phy_device *phydev)
|
||||
+static int at803x_read_specific_status(struct phy_device *phydev,
|
||||
+ struct at803x_ss_mask ss_mask)
|
||||
{
|
||||
int ss;
|
||||
|
||||
@@ -940,11 +946,8 @@ static int at803x_read_specific_status(s
|
||||
if (sfc < 0)
|
||||
return sfc;
|
||||
|
||||
- /* qca8081 takes the different bits for speed value from at803x */
|
||||
- if (phydev->drv->phy_id == QCA8081_PHY_ID)
|
||||
- speed = FIELD_GET(QCA808X_SS_SPEED_MASK, ss);
|
||||
- else
|
||||
- speed = FIELD_GET(AT803X_SS_SPEED_MASK, ss);
|
||||
+ speed = ss & ss_mask.speed_mask;
|
||||
+ speed >>= ss_mask.speed_shift;
|
||||
|
||||
switch (speed) {
|
||||
case AT803X_SS_SPEED_10:
|
||||
@@ -989,6 +992,7 @@ static int at803x_read_specific_status(s
|
||||
static int at803x_read_status(struct phy_device *phydev)
|
||||
{
|
||||
struct at803x_priv *priv = phydev->priv;
|
||||
+ struct at803x_ss_mask ss_mask = { 0 };
|
||||
int err, old_link = phydev->link;
|
||||
|
||||
if (priv->is_1000basex)
|
||||
@@ -1012,7 +1016,9 @@ static int at803x_read_status(struct phy
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
- err = at803x_read_specific_status(phydev);
|
||||
+ ss_mask.speed_mask = AT803X_SS_SPEED_MASK;
|
||||
+ ss_mask.speed_shift = __bf_shf(AT803X_SS_SPEED_MASK);
|
||||
+ err = at803x_read_specific_status(phydev, ss_mask);
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
@@ -1869,6 +1875,7 @@ static int qca808x_config_init(struct ph
|
||||
|
||||
static int qca808x_read_status(struct phy_device *phydev)
|
||||
{
|
||||
+ struct at803x_ss_mask ss_mask = { 0 };
|
||||
int ret;
|
||||
|
||||
ret = phy_read_mmd(phydev, MDIO_MMD_AN, MDIO_AN_10GBT_STAT);
|
||||
@@ -1882,7 +1889,10 @@ static int qca808x_read_status(struct ph
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
- ret = at803x_read_specific_status(phydev);
|
||||
+ /* qca8081 takes the different bits for speed value from at803x */
|
||||
+ ss_mask.speed_mask = QCA808X_SS_SPEED_MASK;
|
||||
+ ss_mask.speed_shift = __bf_shf(QCA808X_SS_SPEED_MASK);
|
||||
+ ret = at803x_read_specific_status(phydev, ss_mask);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
@ -0,0 +1,27 @@
|
||||
From fc9d7264ddc32eaa647d6bfcdc25cdf9f786fde0 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Mon, 18 Dec 2023 00:27:39 +0100
|
||||
Subject: [PATCH 1/2] net: phy: at803x: remove extra space after cast
|
||||
|
||||
Remove extra space after cast as reported by checkpatch to keep code
|
||||
clean.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Link: https://lore.kernel.org/r/20231217232739.27065-1-ansuelsmth@gmail.com
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 2 +-
|
||||
1 file changed, 1 insertion(+), 1 deletion(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -462,7 +462,7 @@ static int at803x_set_wol(struct phy_dev
|
||||
if (!ndev)
|
||||
return -ENODEV;
|
||||
|
||||
- mac = (const u8 *) ndev->dev_addr;
|
||||
+ mac = (const u8 *)ndev->dev_addr;
|
||||
|
||||
if (!is_valid_ether_addr(mac))
|
||||
return -EINVAL;
|
@ -0,0 +1,38 @@
|
||||
From 3ab5720881a924fb6405d9e6a3b09f1026467c47 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Mon, 18 Dec 2023 00:25:08 +0100
|
||||
Subject: [PATCH 2/2] net: phy: at803x: replace msleep(1) with usleep_range
|
||||
|
||||
Replace msleep(1) with usleep_range as suggested by timers-howto guide.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Link: https://lore.kernel.org/r/20231217232508.26470-1-ansuelsmth@gmail.com
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 6 +++---
|
||||
1 file changed, 3 insertions(+), 3 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -916,9 +916,9 @@ static void at803x_link_change_notify(st
|
||||
at803x_context_save(phydev, &context);
|
||||
|
||||
phy_device_reset(phydev, 1);
|
||||
- msleep(1);
|
||||
+ usleep_range(1000, 2000);
|
||||
phy_device_reset(phydev, 0);
|
||||
- msleep(1);
|
||||
+ usleep_range(1000, 2000);
|
||||
|
||||
at803x_context_restore(phydev, &context);
|
||||
|
||||
@@ -1733,7 +1733,7 @@ static int qca83xx_resume(struct phy_dev
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
- msleep(1);
|
||||
+ usleep_range(1000, 2000);
|
||||
|
||||
return 0;
|
||||
}
|
@ -0,0 +1,152 @@
|
||||
From 7961ef1fa10ec35ad6923fb5751877116e4b035b Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 19 Dec 2023 21:21:24 +0100
|
||||
Subject: [PATCH] net: phy: at803x: better align function varibles to open
|
||||
parenthesis
|
||||
|
||||
Better align function variables to open parenthesis as suggested by
|
||||
checkpatch script for qca808x function to make code cleaner.
|
||||
|
||||
For cable_test_get_status function some additional rework was needed to
|
||||
handle too long functions.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 67 ++++++++++++++++++++++------------------
|
||||
1 file changed, 37 insertions(+), 30 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -1781,27 +1781,27 @@ static int qca808x_phy_fast_retrain_conf
|
||||
return ret;
|
||||
|
||||
phy_write_mmd(phydev, MDIO_MMD_AN, QCA808X_PHY_MMD7_TOP_OPTION1,
|
||||
- QCA808X_TOP_OPTION1_DATA);
|
||||
+ QCA808X_TOP_OPTION1_DATA);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PMAPMD, QCA808X_PHY_MMD1_MSE_THRESHOLD_20DB,
|
||||
- QCA808X_MSE_THRESHOLD_20DB_VALUE);
|
||||
+ QCA808X_MSE_THRESHOLD_20DB_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PMAPMD, QCA808X_PHY_MMD1_MSE_THRESHOLD_17DB,
|
||||
- QCA808X_MSE_THRESHOLD_17DB_VALUE);
|
||||
+ QCA808X_MSE_THRESHOLD_17DB_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PMAPMD, QCA808X_PHY_MMD1_MSE_THRESHOLD_27DB,
|
||||
- QCA808X_MSE_THRESHOLD_27DB_VALUE);
|
||||
+ QCA808X_MSE_THRESHOLD_27DB_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PMAPMD, QCA808X_PHY_MMD1_MSE_THRESHOLD_28DB,
|
||||
- QCA808X_MSE_THRESHOLD_28DB_VALUE);
|
||||
+ QCA808X_MSE_THRESHOLD_28DB_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PCS, QCA808X_PHY_MMD3_DEBUG_1,
|
||||
- QCA808X_MMD3_DEBUG_1_VALUE);
|
||||
+ QCA808X_MMD3_DEBUG_1_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PCS, QCA808X_PHY_MMD3_DEBUG_4,
|
||||
- QCA808X_MMD3_DEBUG_4_VALUE);
|
||||
+ QCA808X_MMD3_DEBUG_4_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PCS, QCA808X_PHY_MMD3_DEBUG_5,
|
||||
- QCA808X_MMD3_DEBUG_5_VALUE);
|
||||
+ QCA808X_MMD3_DEBUG_5_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PCS, QCA808X_PHY_MMD3_DEBUG_3,
|
||||
- QCA808X_MMD3_DEBUG_3_VALUE);
|
||||
+ QCA808X_MMD3_DEBUG_3_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PCS, QCA808X_PHY_MMD3_DEBUG_6,
|
||||
- QCA808X_MMD3_DEBUG_6_VALUE);
|
||||
+ QCA808X_MMD3_DEBUG_6_VALUE);
|
||||
phy_write_mmd(phydev, MDIO_MMD_PCS, QCA808X_PHY_MMD3_DEBUG_2,
|
||||
- QCA808X_MMD3_DEBUG_2_VALUE);
|
||||
+ QCA808X_MMD3_DEBUG_2_VALUE);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -1838,13 +1838,14 @@ static int qca808x_config_init(struct ph
|
||||
|
||||
/* Active adc&vga on 802.3az for the link 1000M and 100M */
|
||||
ret = phy_modify_mmd(phydev, MDIO_MMD_PCS, QCA808X_PHY_MMD3_ADDR_CLD_CTRL7,
|
||||
- QCA808X_8023AZ_AFE_CTRL_MASK, QCA808X_8023AZ_AFE_EN);
|
||||
+ QCA808X_8023AZ_AFE_CTRL_MASK, QCA808X_8023AZ_AFE_EN);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* Adjust the threshold on 802.3az for the link 1000M */
|
||||
ret = phy_write_mmd(phydev, MDIO_MMD_PCS,
|
||||
- QCA808X_PHY_MMD3_AZ_TRAINING_CTRL, QCA808X_MMD3_AZ_TRAINING_VAL);
|
||||
+ QCA808X_PHY_MMD3_AZ_TRAINING_CTRL,
|
||||
+ QCA808X_MMD3_AZ_TRAINING_VAL);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@@ -1870,7 +1871,8 @@ static int qca808x_config_init(struct ph
|
||||
|
||||
/* Configure adc threshold as 100mv for the link 10M */
|
||||
return at803x_debug_reg_mask(phydev, QCA808X_PHY_DEBUG_ADC_THRESHOLD,
|
||||
- QCA808X_ADC_THRESHOLD_MASK, QCA808X_ADC_THRESHOLD_100MV);
|
||||
+ QCA808X_ADC_THRESHOLD_MASK,
|
||||
+ QCA808X_ADC_THRESHOLD_100MV);
|
||||
}
|
||||
|
||||
static int qca808x_read_status(struct phy_device *phydev)
|
||||
@@ -1883,7 +1885,7 @@ static int qca808x_read_status(struct ph
|
||||
return ret;
|
||||
|
||||
linkmode_mod_bit(ETHTOOL_LINK_MODE_2500baseT_Full_BIT, phydev->lp_advertising,
|
||||
- ret & MDIO_AN_10GBT_STAT_LP2_5G);
|
||||
+ ret & MDIO_AN_10GBT_STAT_LP2_5G);
|
||||
|
||||
ret = genphy_read_status(phydev);
|
||||
if (ret)
|
||||
@@ -1913,7 +1915,7 @@ static int qca808x_read_status(struct ph
|
||||
*/
|
||||
if (qca808x_has_fast_retrain_or_slave_seed(phydev)) {
|
||||
if (phydev->master_slave_state == MASTER_SLAVE_STATE_ERR ||
|
||||
- qca808x_is_prefer_master(phydev)) {
|
||||
+ qca808x_is_prefer_master(phydev)) {
|
||||
qca808x_phy_ms_seed_enable(phydev, false);
|
||||
} else {
|
||||
qca808x_phy_ms_seed_enable(phydev, true);
|
||||
@@ -2070,18 +2072,22 @@ static int qca808x_cable_test_get_status
|
||||
ethnl_cable_test_result(phydev, ETHTOOL_A_CABLE_PAIR_D,
|
||||
qca808x_cable_test_result_trans(pair_d));
|
||||
|
||||
- if (qca808x_cdt_fault_length_valid(pair_a))
|
||||
- ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_A,
|
||||
- qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_A));
|
||||
- if (qca808x_cdt_fault_length_valid(pair_b))
|
||||
- ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_B,
|
||||
- qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_B));
|
||||
- if (qca808x_cdt_fault_length_valid(pair_c))
|
||||
- ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_C,
|
||||
- qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_C));
|
||||
- if (qca808x_cdt_fault_length_valid(pair_d))
|
||||
- ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_D,
|
||||
- qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_D));
|
||||
+ if (qca808x_cdt_fault_length_valid(pair_a)) {
|
||||
+ val = qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_A);
|
||||
+ ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_A, val);
|
||||
+ }
|
||||
+ if (qca808x_cdt_fault_length_valid(pair_b)) {
|
||||
+ val = qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_B);
|
||||
+ ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_B, val);
|
||||
+ }
|
||||
+ if (qca808x_cdt_fault_length_valid(pair_c)) {
|
||||
+ val = qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_C);
|
||||
+ ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_C, val);
|
||||
+ }
|
||||
+ if (qca808x_cdt_fault_length_valid(pair_d)) {
|
||||
+ val = qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_D);
|
||||
+ ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_D, val);
|
||||
+ }
|
||||
|
||||
*finished = true;
|
||||
|
||||
@@ -2148,8 +2154,9 @@ static void qca808x_link_change_notify(s
|
||||
* the interface device address is always phy address added by 1.
|
||||
*/
|
||||
mdiobus_c45_modify_changed(phydev->mdio.bus, phydev->mdio.addr + 1,
|
||||
- MDIO_MMD_PMAPMD, QCA8081_PHY_SERDES_MMD1_FIFO_CTRL,
|
||||
- QCA8081_PHY_FIFO_RSTN, phydev->link ? QCA8081_PHY_FIFO_RSTN : 0);
|
||||
+ MDIO_MMD_PMAPMD, QCA8081_PHY_SERDES_MMD1_FIFO_CTRL,
|
||||
+ QCA8081_PHY_FIFO_RSTN,
|
||||
+ phydev->link ? QCA8081_PHY_FIFO_RSTN : 0);
|
||||
}
|
||||
|
||||
static struct phy_driver at803x_driver[] = {
|
@ -0,0 +1,62 @@
|
||||
From 22eb276098da820d9440fad22901f9b74ed4d659 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 4 Jan 2024 22:30:38 +0100
|
||||
Subject: [PATCH 1/4] net: phy: at803x: generalize cdt fault length function
|
||||
|
||||
Generalize cable test fault length function since they all base on the
|
||||
same magic values (already reverse engineered to understand the meaning
|
||||
of it) to have consistenct values on every PHY.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Simon Horman <horms@kernel.org>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 13 ++++++-------
|
||||
1 file changed, 6 insertions(+), 7 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -1192,10 +1192,8 @@ static bool at803x_cdt_fault_length_vali
|
||||
return false;
|
||||
}
|
||||
|
||||
-static int at803x_cdt_fault_length(u16 status)
|
||||
+static int at803x_cdt_fault_length(int dt)
|
||||
{
|
||||
- int dt;
|
||||
-
|
||||
/* According to the datasheet the distance to the fault is
|
||||
* DELTA_TIME * 0.824 meters.
|
||||
*
|
||||
@@ -1211,8 +1209,6 @@ static int at803x_cdt_fault_length(u16 s
|
||||
* With a VF of 0.69 we get the factor 0.824 mentioned in the
|
||||
* datasheet.
|
||||
*/
|
||||
- dt = FIELD_GET(AT803X_CDT_STATUS_DELTA_TIME_MASK, status);
|
||||
-
|
||||
return (dt * 824) / 10;
|
||||
}
|
||||
|
||||
@@ -1265,9 +1261,11 @@ static int at803x_cable_test_one_pair(st
|
||||
ethnl_cable_test_result(phydev, ethtool_pair[pair],
|
||||
at803x_cable_test_result_trans(val));
|
||||
|
||||
- if (at803x_cdt_fault_length_valid(val))
|
||||
+ if (at803x_cdt_fault_length_valid(val)) {
|
||||
+ val = FIELD_GET(AT803X_CDT_STATUS_DELTA_TIME_MASK, val);
|
||||
ethnl_cable_test_fault_length(phydev, ethtool_pair[pair],
|
||||
at803x_cdt_fault_length(val));
|
||||
+ }
|
||||
|
||||
return 1;
|
||||
}
|
||||
@@ -1992,7 +1990,8 @@ static int qca808x_cdt_fault_length(stru
|
||||
if (val < 0)
|
||||
return val;
|
||||
|
||||
- return (FIELD_GET(QCA808X_CDT_DIAG_LENGTH, val) * 824) / 10;
|
||||
+ val = FIELD_GET(QCA808X_CDT_DIAG_LENGTH, val);
|
||||
+ return at803x_cdt_fault_length(val);
|
||||
}
|
||||
|
||||
static int qca808x_cable_test_start(struct phy_device *phydev)
|
@ -0,0 +1,118 @@
|
||||
From e0e9ada1df6133513249861c1d91c1dbefd9383b Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 4 Jan 2024 22:30:39 +0100
|
||||
Subject: [PATCH 2/4] net: phy: at803x: refactor qca808x cable test get status
|
||||
function
|
||||
|
||||
Refactor qca808x cable test get status function to remove code
|
||||
duplication and clean things up.
|
||||
|
||||
The same logic is applied to each pair hence it can be generalized and
|
||||
moved to a common function.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Simon Horman <horms@kernel.org>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 80 ++++++++++++++++++++++++----------------
|
||||
1 file changed, 49 insertions(+), 31 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -2035,10 +2035,43 @@ static int qca808x_cable_test_start(stru
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int qca808x_cable_test_get_pair_status(struct phy_device *phydev, u8 pair,
|
||||
+ u16 status)
|
||||
+{
|
||||
+ u16 pair_code;
|
||||
+ int length;
|
||||
+
|
||||
+ switch (pair) {
|
||||
+ case ETHTOOL_A_CABLE_PAIR_A:
|
||||
+ pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_A, status);
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_B:
|
||||
+ pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_B, status);
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_C:
|
||||
+ pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_C, status);
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_D:
|
||||
+ pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_D, status);
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ ethnl_cable_test_result(phydev, pair,
|
||||
+ qca808x_cable_test_result_trans(pair_code));
|
||||
+
|
||||
+ if (qca808x_cdt_fault_length_valid(pair_code)) {
|
||||
+ length = qca808x_cdt_fault_length(phydev, pair);
|
||||
+ ethnl_cable_test_fault_length(phydev, pair, length);
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
static int qca808x_cable_test_get_status(struct phy_device *phydev, bool *finished)
|
||||
{
|
||||
int ret, val;
|
||||
- int pair_a, pair_b, pair_c, pair_d;
|
||||
|
||||
*finished = false;
|
||||
|
||||
@@ -2057,36 +2090,21 @@ static int qca808x_cable_test_get_status
|
||||
if (val < 0)
|
||||
return val;
|
||||
|
||||
- pair_a = FIELD_GET(QCA808X_CDT_CODE_PAIR_A, val);
|
||||
- pair_b = FIELD_GET(QCA808X_CDT_CODE_PAIR_B, val);
|
||||
- pair_c = FIELD_GET(QCA808X_CDT_CODE_PAIR_C, val);
|
||||
- pair_d = FIELD_GET(QCA808X_CDT_CODE_PAIR_D, val);
|
||||
-
|
||||
- ethnl_cable_test_result(phydev, ETHTOOL_A_CABLE_PAIR_A,
|
||||
- qca808x_cable_test_result_trans(pair_a));
|
||||
- ethnl_cable_test_result(phydev, ETHTOOL_A_CABLE_PAIR_B,
|
||||
- qca808x_cable_test_result_trans(pair_b));
|
||||
- ethnl_cable_test_result(phydev, ETHTOOL_A_CABLE_PAIR_C,
|
||||
- qca808x_cable_test_result_trans(pair_c));
|
||||
- ethnl_cable_test_result(phydev, ETHTOOL_A_CABLE_PAIR_D,
|
||||
- qca808x_cable_test_result_trans(pair_d));
|
||||
-
|
||||
- if (qca808x_cdt_fault_length_valid(pair_a)) {
|
||||
- val = qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_A);
|
||||
- ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_A, val);
|
||||
- }
|
||||
- if (qca808x_cdt_fault_length_valid(pair_b)) {
|
||||
- val = qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_B);
|
||||
- ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_B, val);
|
||||
- }
|
||||
- if (qca808x_cdt_fault_length_valid(pair_c)) {
|
||||
- val = qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_C);
|
||||
- ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_C, val);
|
||||
- }
|
||||
- if (qca808x_cdt_fault_length_valid(pair_d)) {
|
||||
- val = qca808x_cdt_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_D);
|
||||
- ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_D, val);
|
||||
- }
|
||||
+ ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_A, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_B, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_C, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_D, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
|
||||
*finished = true;
|
||||
|
@ -0,0 +1,182 @@
|
||||
From ea73e5ea442ee2aade67b1fb1233ccb3cbea2ceb Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 4 Jan 2024 22:30:40 +0100
|
||||
Subject: [PATCH 3/4] net: phy: at803x: add support for cdt cross short test
|
||||
for qca808x
|
||||
|
||||
QCA808x PHY Family supports Cable Diagnostic Test also for Cross Pair
|
||||
Short.
|
||||
|
||||
Add all the define to make enable and support these additional tests.
|
||||
|
||||
Cross Short test was previously disabled by default, this is now changed
|
||||
and enabled by default. In this mode, the mask changed a bit and length
|
||||
is shifted based on the fault condition.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Simon Horman <horms@kernel.org>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 86 ++++++++++++++++++++++++++++++++--------
|
||||
1 file changed, 69 insertions(+), 17 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -254,6 +254,7 @@
|
||||
|
||||
#define QCA808X_CDT_ENABLE_TEST BIT(15)
|
||||
#define QCA808X_CDT_INTER_CHECK_DIS BIT(13)
|
||||
+#define QCA808X_CDT_STATUS BIT(11)
|
||||
#define QCA808X_CDT_LENGTH_UNIT BIT(10)
|
||||
|
||||
#define QCA808X_MMD3_CDT_STATUS 0x8064
|
||||
@@ -261,16 +262,44 @@
|
||||
#define QCA808X_MMD3_CDT_DIAG_PAIR_B 0x8066
|
||||
#define QCA808X_MMD3_CDT_DIAG_PAIR_C 0x8067
|
||||
#define QCA808X_MMD3_CDT_DIAG_PAIR_D 0x8068
|
||||
-#define QCA808X_CDT_DIAG_LENGTH GENMASK(7, 0)
|
||||
+#define QCA808X_CDT_DIAG_LENGTH_SAME_SHORT GENMASK(15, 8)
|
||||
+#define QCA808X_CDT_DIAG_LENGTH_CROSS_SHORT GENMASK(7, 0)
|
||||
|
||||
#define QCA808X_CDT_CODE_PAIR_A GENMASK(15, 12)
|
||||
#define QCA808X_CDT_CODE_PAIR_B GENMASK(11, 8)
|
||||
#define QCA808X_CDT_CODE_PAIR_C GENMASK(7, 4)
|
||||
#define QCA808X_CDT_CODE_PAIR_D GENMASK(3, 0)
|
||||
-#define QCA808X_CDT_STATUS_STAT_FAIL 0
|
||||
-#define QCA808X_CDT_STATUS_STAT_NORMAL 1
|
||||
-#define QCA808X_CDT_STATUS_STAT_OPEN 2
|
||||
-#define QCA808X_CDT_STATUS_STAT_SHORT 3
|
||||
+
|
||||
+#define QCA808X_CDT_STATUS_STAT_TYPE GENMASK(1, 0)
|
||||
+#define QCA808X_CDT_STATUS_STAT_FAIL FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 0)
|
||||
+#define QCA808X_CDT_STATUS_STAT_NORMAL FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 1)
|
||||
+#define QCA808X_CDT_STATUS_STAT_SAME_OPEN FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_SAME_SHORT FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 3)
|
||||
+
|
||||
+#define QCA808X_CDT_STATUS_STAT_MDI GENMASK(3, 2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_MDI1 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 1)
|
||||
+#define QCA808X_CDT_STATUS_STAT_MDI2 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_MDI3 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 3)
|
||||
+
|
||||
+/* NORMAL are MDI with type set to 0 */
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI1
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI1)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI1)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI2
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI3
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI3)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI3)
|
||||
+
|
||||
+/* Added for reference of existence but should be handled by wait_for_completion already */
|
||||
+#define QCA808X_CDT_STATUS_STAT_BUSY (BIT(1) | BIT(3))
|
||||
|
||||
/* QCA808X 1G chip type */
|
||||
#define QCA808X_PHY_MMD7_CHIP_TYPE 0x901d
|
||||
@@ -1941,8 +1970,17 @@ static int qca808x_soft_reset(struct phy
|
||||
static bool qca808x_cdt_fault_length_valid(int cdt_code)
|
||||
{
|
||||
switch (cdt_code) {
|
||||
- case QCA808X_CDT_STATUS_STAT_SHORT:
|
||||
- case QCA808X_CDT_STATUS_STAT_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT:
|
||||
return true;
|
||||
default:
|
||||
return false;
|
||||
@@ -1954,17 +1992,28 @@ static int qca808x_cable_test_result_tra
|
||||
switch (cdt_code) {
|
||||
case QCA808X_CDT_STATUS_STAT_NORMAL:
|
||||
return ETHTOOL_A_CABLE_RESULT_CODE_OK;
|
||||
- case QCA808X_CDT_STATUS_STAT_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_SAME_SHORT:
|
||||
return ETHTOOL_A_CABLE_RESULT_CODE_SAME_SHORT;
|
||||
- case QCA808X_CDT_STATUS_STAT_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_SAME_OPEN:
|
||||
return ETHTOOL_A_CABLE_RESULT_CODE_OPEN;
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT:
|
||||
+ return ETHTOOL_A_CABLE_RESULT_CODE_CROSS_SHORT;
|
||||
case QCA808X_CDT_STATUS_STAT_FAIL:
|
||||
default:
|
||||
return ETHTOOL_A_CABLE_RESULT_CODE_UNSPEC;
|
||||
}
|
||||
}
|
||||
|
||||
-static int qca808x_cdt_fault_length(struct phy_device *phydev, int pair)
|
||||
+static int qca808x_cdt_fault_length(struct phy_device *phydev, int pair,
|
||||
+ int result)
|
||||
{
|
||||
int val;
|
||||
u32 cdt_length_reg = 0;
|
||||
@@ -1990,7 +2039,11 @@ static int qca808x_cdt_fault_length(stru
|
||||
if (val < 0)
|
||||
return val;
|
||||
|
||||
- val = FIELD_GET(QCA808X_CDT_DIAG_LENGTH, val);
|
||||
+ if (result == ETHTOOL_A_CABLE_RESULT_CODE_SAME_SHORT)
|
||||
+ val = FIELD_GET(QCA808X_CDT_DIAG_LENGTH_SAME_SHORT, val);
|
||||
+ else
|
||||
+ val = FIELD_GET(QCA808X_CDT_DIAG_LENGTH_CROSS_SHORT, val);
|
||||
+
|
||||
return at803x_cdt_fault_length(val);
|
||||
}
|
||||
|
||||
@@ -2038,8 +2091,8 @@ static int qca808x_cable_test_start(stru
|
||||
static int qca808x_cable_test_get_pair_status(struct phy_device *phydev, u8 pair,
|
||||
u16 status)
|
||||
{
|
||||
+ int length, result;
|
||||
u16 pair_code;
|
||||
- int length;
|
||||
|
||||
switch (pair) {
|
||||
case ETHTOOL_A_CABLE_PAIR_A:
|
||||
@@ -2058,11 +2111,11 @@ static int qca808x_cable_test_get_pair_s
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
- ethnl_cable_test_result(phydev, pair,
|
||||
- qca808x_cable_test_result_trans(pair_code));
|
||||
+ result = qca808x_cable_test_result_trans(pair_code);
|
||||
+ ethnl_cable_test_result(phydev, pair, result);
|
||||
|
||||
if (qca808x_cdt_fault_length_valid(pair_code)) {
|
||||
- length = qca808x_cdt_fault_length(phydev, pair);
|
||||
+ length = qca808x_cdt_fault_length(phydev, pair, result);
|
||||
ethnl_cable_test_fault_length(phydev, pair, length);
|
||||
}
|
||||
|
||||
@@ -2076,8 +2129,7 @@ static int qca808x_cable_test_get_status
|
||||
*finished = false;
|
||||
|
||||
val = QCA808X_CDT_ENABLE_TEST |
|
||||
- QCA808X_CDT_LENGTH_UNIT |
|
||||
- QCA808X_CDT_INTER_CHECK_DIS;
|
||||
+ QCA808X_CDT_LENGTH_UNIT;
|
||||
ret = at803x_cdt_start(phydev, val);
|
||||
if (ret)
|
||||
return ret;
|
@ -0,0 +1,62 @@
|
||||
From c34d9452d4e5d98a655d7b625e85466320885416 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 4 Jan 2024 22:30:41 +0100
|
||||
Subject: [PATCH 4/4] net: phy: at803x: make read_status more generic
|
||||
|
||||
Make read_status more generic in preparation on moving it to shared
|
||||
library as other PHY Family Driver will have the exact same
|
||||
implementation.
|
||||
|
||||
The only specific part was a check for AR8031/33 if 1000basex was used.
|
||||
The check is moved to a dedicated function specific for those PHYs.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Simon Horman <horms@kernel.org>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 17 ++++++++++++-----
|
||||
1 file changed, 12 insertions(+), 5 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -1020,13 +1020,9 @@ static int at803x_read_specific_status(s
|
||||
|
||||
static int at803x_read_status(struct phy_device *phydev)
|
||||
{
|
||||
- struct at803x_priv *priv = phydev->priv;
|
||||
struct at803x_ss_mask ss_mask = { 0 };
|
||||
int err, old_link = phydev->link;
|
||||
|
||||
- if (priv->is_1000basex)
|
||||
- return genphy_c37_read_status(phydev);
|
||||
-
|
||||
/* Update the link, but return if there was an error */
|
||||
err = genphy_update_link(phydev);
|
||||
if (err)
|
||||
@@ -1618,6 +1614,17 @@ static int at8031_config_intr(struct phy
|
||||
return at803x_config_intr(phydev);
|
||||
}
|
||||
|
||||
+/* AR8031 and AR8033 share the same read status logic */
|
||||
+static int at8031_read_status(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+
|
||||
+ if (priv->is_1000basex)
|
||||
+ return genphy_c37_read_status(phydev);
|
||||
+
|
||||
+ return at803x_read_status(phydev);
|
||||
+}
|
||||
+
|
||||
/* AR8031 and AR8035 share the same cable test get status reg */
|
||||
static int at8031_cable_test_get_status(struct phy_device *phydev,
|
||||
bool *finished)
|
||||
@@ -2281,7 +2288,7 @@ static struct phy_driver at803x_driver[]
|
||||
.read_page = at803x_read_page,
|
||||
.write_page = at803x_write_page,
|
||||
.get_features = at803x_get_features,
|
||||
- .read_status = at803x_read_status,
|
||||
+ .read_status = at8031_read_status,
|
||||
.config_intr = at8031_config_intr,
|
||||
.handle_interrupt = at803x_handle_interrupt,
|
||||
.get_tunable = at803x_get_tunable,
|
@ -0,0 +1,408 @@
|
||||
From 7196062b64ee470b91015f3d2e82d225948258ea Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 25 Jan 2024 21:37:01 +0100
|
||||
Subject: [PATCH 5/5] net: phy: at803x: add LED support for qca808x
|
||||
|
||||
Add LED support for QCA8081 PHY.
|
||||
|
||||
Documentation for this LEDs PHY is very scarce even with NDA access
|
||||
to Documentation for OEMs. Only the blink pattern are documented and are
|
||||
very confusing most of the time. No documentation is present about
|
||||
forcing the LED on/off or to always blink.
|
||||
|
||||
Those settings were reversed by poking the regs and trying to find the
|
||||
correct bits to trigger these modes. Some bits mode are not clear and
|
||||
maybe the documentation option are not 100% correct. For the sake of LED
|
||||
support the reversed option are enough to add support for current LED
|
||||
APIs.
|
||||
|
||||
Supported HW control modes are:
|
||||
- tx
|
||||
- rx
|
||||
- link_10
|
||||
- link_100
|
||||
- link_1000
|
||||
- link_2500
|
||||
- half_duplex
|
||||
- full_duplex
|
||||
|
||||
Also add support for LED polarity set to set LED polarity to active
|
||||
high or low. QSDK sets this value to high by default but PHY reset value
|
||||
doesn't have this enabled by default.
|
||||
|
||||
QSDK also sets 2 additional bits but their usage is not clear, info about
|
||||
this is added in the header. It was verified that for correct function
|
||||
of the LED if active high is needed, only BIT 6 is needed.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Link: https://lore.kernel.org/r/20240125203702.4552-6-ansuelsmth@gmail.com
|
||||
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
|
||||
---
|
||||
drivers/net/phy/at803x.c | 327 +++++++++++++++++++++++++++++++++++++++
|
||||
1 file changed, 327 insertions(+)
|
||||
|
||||
--- a/drivers/net/phy/at803x.c
|
||||
+++ b/drivers/net/phy/at803x.c
|
||||
@@ -301,6 +301,87 @@
|
||||
/* Added for reference of existence but should be handled by wait_for_completion already */
|
||||
#define QCA808X_CDT_STATUS_STAT_BUSY (BIT(1) | BIT(3))
|
||||
|
||||
+#define QCA808X_MMD7_LED_GLOBAL 0x8073
|
||||
+#define QCA808X_LED_BLINK_1 GENMASK(11, 6)
|
||||
+#define QCA808X_LED_BLINK_2 GENMASK(5, 0)
|
||||
+/* Values are the same for both BLINK_1 and BLINK_2 */
|
||||
+#define QCA808X_LED_BLINK_FREQ_MASK GENMASK(5, 3)
|
||||
+#define QCA808X_LED_BLINK_FREQ_2HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x0)
|
||||
+#define QCA808X_LED_BLINK_FREQ_4HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x1)
|
||||
+#define QCA808X_LED_BLINK_FREQ_8HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x2)
|
||||
+#define QCA808X_LED_BLINK_FREQ_16HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x3)
|
||||
+#define QCA808X_LED_BLINK_FREQ_32HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x4)
|
||||
+#define QCA808X_LED_BLINK_FREQ_64HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x5)
|
||||
+#define QCA808X_LED_BLINK_FREQ_128HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x6)
|
||||
+#define QCA808X_LED_BLINK_FREQ_256HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x7)
|
||||
+#define QCA808X_LED_BLINK_DUTY_MASK GENMASK(2, 0)
|
||||
+#define QCA808X_LED_BLINK_DUTY_50_50 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x0)
|
||||
+#define QCA808X_LED_BLINK_DUTY_75_25 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x1)
|
||||
+#define QCA808X_LED_BLINK_DUTY_25_75 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x2)
|
||||
+#define QCA808X_LED_BLINK_DUTY_33_67 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x3)
|
||||
+#define QCA808X_LED_BLINK_DUTY_67_33 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x4)
|
||||
+#define QCA808X_LED_BLINK_DUTY_17_83 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x5)
|
||||
+#define QCA808X_LED_BLINK_DUTY_83_17 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x6)
|
||||
+#define QCA808X_LED_BLINK_DUTY_8_92 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x7)
|
||||
+
|
||||
+#define QCA808X_MMD7_LED2_CTRL 0x8074
|
||||
+#define QCA808X_MMD7_LED2_FORCE_CTRL 0x8075
|
||||
+#define QCA808X_MMD7_LED1_CTRL 0x8076
|
||||
+#define QCA808X_MMD7_LED1_FORCE_CTRL 0x8077
|
||||
+#define QCA808X_MMD7_LED0_CTRL 0x8078
|
||||
+#define QCA808X_MMD7_LED_CTRL(x) (0x8078 - ((x) * 2))
|
||||
+
|
||||
+/* LED hw control pattern is the same for every LED */
|
||||
+#define QCA808X_LED_PATTERN_MASK GENMASK(15, 0)
|
||||
+#define QCA808X_LED_SPEED2500_ON BIT(15)
|
||||
+#define QCA808X_LED_SPEED2500_BLINK BIT(14)
|
||||
+/* Follow blink trigger even if duplex or speed condition doesn't match */
|
||||
+#define QCA808X_LED_BLINK_CHECK_BYPASS BIT(13)
|
||||
+#define QCA808X_LED_FULL_DUPLEX_ON BIT(12)
|
||||
+#define QCA808X_LED_HALF_DUPLEX_ON BIT(11)
|
||||
+#define QCA808X_LED_TX_BLINK BIT(10)
|
||||
+#define QCA808X_LED_RX_BLINK BIT(9)
|
||||
+#define QCA808X_LED_TX_ON_10MS BIT(8)
|
||||
+#define QCA808X_LED_RX_ON_10MS BIT(7)
|
||||
+#define QCA808X_LED_SPEED1000_ON BIT(6)
|
||||
+#define QCA808X_LED_SPEED100_ON BIT(5)
|
||||
+#define QCA808X_LED_SPEED10_ON BIT(4)
|
||||
+#define QCA808X_LED_COLLISION_BLINK BIT(3)
|
||||
+#define QCA808X_LED_SPEED1000_BLINK BIT(2)
|
||||
+#define QCA808X_LED_SPEED100_BLINK BIT(1)
|
||||
+#define QCA808X_LED_SPEED10_BLINK BIT(0)
|
||||
+
|
||||
+#define QCA808X_MMD7_LED0_FORCE_CTRL 0x8079
|
||||
+#define QCA808X_MMD7_LED_FORCE_CTRL(x) (0x8079 - ((x) * 2))
|
||||
+
|
||||
+/* LED force ctrl is the same for every LED
|
||||
+ * No documentation exist for this, not even internal one
|
||||
+ * with NDA as QCOM gives only info about configuring
|
||||
+ * hw control pattern rules and doesn't indicate any way
|
||||
+ * to force the LED to specific mode.
|
||||
+ * These define comes from reverse and testing and maybe
|
||||
+ * lack of some info or some info are not entirely correct.
|
||||
+ * For the basic LED control and hw control these finding
|
||||
+ * are enough to support LED control in all the required APIs.
|
||||
+ *
|
||||
+ * On doing some comparison with implementation with qca807x,
|
||||
+ * it was found that it's 1:1 equal to it and confirms all the
|
||||
+ * reverse done. It was also found further specification with the
|
||||
+ * force mode and the blink modes.
|
||||
+ */
|
||||
+#define QCA808X_LED_FORCE_EN BIT(15)
|
||||
+#define QCA808X_LED_FORCE_MODE_MASK GENMASK(14, 13)
|
||||
+#define QCA808X_LED_FORCE_BLINK_1 FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x3)
|
||||
+#define QCA808X_LED_FORCE_BLINK_2 FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x2)
|
||||
+#define QCA808X_LED_FORCE_ON FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x1)
|
||||
+#define QCA808X_LED_FORCE_OFF FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x0)
|
||||
+
|
||||
+#define QCA808X_MMD7_LED_POLARITY_CTRL 0x901a
|
||||
+/* QSDK sets by default 0x46 to this reg that sets BIT 6 for
|
||||
+ * LED to active high. It's not clear what BIT 3 and BIT 4 does.
|
||||
+ */
|
||||
+#define QCA808X_LED_ACTIVE_HIGH BIT(6)
|
||||
+
|
||||
/* QCA808X 1G chip type */
|
||||
#define QCA808X_PHY_MMD7_CHIP_TYPE 0x901d
|
||||
#define QCA808X_PHY_CHIP_TYPE_1G BIT(0)
|
||||
@@ -346,6 +427,7 @@ struct at803x_priv {
|
||||
struct regulator_dev *vddio_rdev;
|
||||
struct regulator_dev *vddh_rdev;
|
||||
u64 stats[ARRAY_SIZE(qca83xx_hw_stats)];
|
||||
+ int led_polarity_mode;
|
||||
};
|
||||
|
||||
struct at803x_context {
|
||||
@@ -706,6 +788,9 @@ static int at803x_probe(struct phy_devic
|
||||
if (!priv)
|
||||
return -ENOMEM;
|
||||
|
||||
+ /* Init LED polarity mode to -1 */
|
||||
+ priv->led_polarity_mode = -1;
|
||||
+
|
||||
phydev->priv = priv;
|
||||
|
||||
ret = at803x_parse_dt(phydev);
|
||||
@@ -2235,6 +2320,242 @@ static void qca808x_link_change_notify(s
|
||||
phydev->link ? QCA8081_PHY_FIFO_RSTN : 0);
|
||||
}
|
||||
|
||||
+static int qca808x_led_parse_netdev(struct phy_device *phydev, unsigned long rules,
|
||||
+ u16 *offload_trigger)
|
||||
+{
|
||||
+ /* Parsing specific to netdev trigger */
|
||||
+ if (test_bit(TRIGGER_NETDEV_TX, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_TX_BLINK;
|
||||
+ if (test_bit(TRIGGER_NETDEV_RX, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_RX_BLINK;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_10, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_SPEED10_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_100, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_SPEED100_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_1000, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_SPEED1000_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_2500, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_SPEED2500_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_HALF_DUPLEX, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_HALF_DUPLEX_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_FULL_DUPLEX, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_FULL_DUPLEX_ON;
|
||||
+
|
||||
+ if (rules && !*offload_trigger)
|
||||
+ return -EOPNOTSUPP;
|
||||
+
|
||||
+ /* Enable BLINK_CHECK_BYPASS by default to make the LED
|
||||
+ * blink even with duplex or speed mode not enabled.
|
||||
+ */
|
||||
+ *offload_trigger |= QCA808X_LED_BLINK_CHECK_BYPASS;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca808x_led_hw_control_enable(struct phy_device *phydev, u8 index)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+
|
||||
+ if (index > 2)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ reg = QCA808X_MMD7_LED_FORCE_CTRL(index);
|
||||
+
|
||||
+ return phy_clear_bits_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
+ QCA808X_LED_FORCE_EN);
|
||||
+}
|
||||
+
|
||||
+static int qca808x_led_hw_is_supported(struct phy_device *phydev, u8 index,
|
||||
+ unsigned long rules)
|
||||
+{
|
||||
+ u16 offload_trigger = 0;
|
||||
+
|
||||
+ if (index > 2)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ return qca808x_led_parse_netdev(phydev, rules, &offload_trigger);
|
||||
+}
|
||||
+
|
||||
+static int qca808x_led_hw_control_set(struct phy_device *phydev, u8 index,
|
||||
+ unsigned long rules)
|
||||
+{
|
||||
+ u16 reg, offload_trigger = 0;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (index > 2)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ reg = QCA808X_MMD7_LED_CTRL(index);
|
||||
+
|
||||
+ ret = qca808x_led_parse_netdev(phydev, rules, &offload_trigger);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = qca808x_led_hw_control_enable(phydev, index);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ return phy_modify_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
+ QCA808X_LED_PATTERN_MASK,
|
||||
+ offload_trigger);
|
||||
+}
|
||||
+
|
||||
+static bool qca808x_led_hw_control_status(struct phy_device *phydev, u8 index)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+ int val;
|
||||
+
|
||||
+ if (index > 2)
|
||||
+ return false;
|
||||
+
|
||||
+ reg = QCA808X_MMD7_LED_FORCE_CTRL(index);
|
||||
+
|
||||
+ val = phy_read_mmd(phydev, MDIO_MMD_AN, reg);
|
||||
+
|
||||
+ return !(val & QCA808X_LED_FORCE_EN);
|
||||
+}
|
||||
+
|
||||
+static int qca808x_led_hw_control_get(struct phy_device *phydev, u8 index,
|
||||
+ unsigned long *rules)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+ int val;
|
||||
+
|
||||
+ if (index > 2)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ /* Check if we have hw control enabled */
|
||||
+ if (qca808x_led_hw_control_status(phydev, index))
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ reg = QCA808X_MMD7_LED_CTRL(index);
|
||||
+
|
||||
+ val = phy_read_mmd(phydev, MDIO_MMD_AN, reg);
|
||||
+ if (val & QCA808X_LED_TX_BLINK)
|
||||
+ set_bit(TRIGGER_NETDEV_TX, rules);
|
||||
+ if (val & QCA808X_LED_RX_BLINK)
|
||||
+ set_bit(TRIGGER_NETDEV_RX, rules);
|
||||
+ if (val & QCA808X_LED_SPEED10_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_10, rules);
|
||||
+ if (val & QCA808X_LED_SPEED100_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_100, rules);
|
||||
+ if (val & QCA808X_LED_SPEED1000_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_1000, rules);
|
||||
+ if (val & QCA808X_LED_SPEED2500_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_2500, rules);
|
||||
+ if (val & QCA808X_LED_HALF_DUPLEX_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_HALF_DUPLEX, rules);
|
||||
+ if (val & QCA808X_LED_FULL_DUPLEX_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_FULL_DUPLEX, rules);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca808x_led_hw_control_reset(struct phy_device *phydev, u8 index)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+
|
||||
+ if (index > 2)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ reg = QCA808X_MMD7_LED_CTRL(index);
|
||||
+
|
||||
+ return phy_clear_bits_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
+ QCA808X_LED_PATTERN_MASK);
|
||||
+}
|
||||
+
|
||||
+static int qca808x_led_brightness_set(struct phy_device *phydev,
|
||||
+ u8 index, enum led_brightness value)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (index > 2)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ if (!value) {
|
||||
+ ret = qca808x_led_hw_control_reset(phydev, index);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ reg = QCA808X_MMD7_LED_FORCE_CTRL(index);
|
||||
+
|
||||
+ return phy_modify_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
+ QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_MODE_MASK,
|
||||
+ QCA808X_LED_FORCE_EN | value ? QCA808X_LED_FORCE_ON :
|
||||
+ QCA808X_LED_FORCE_OFF);
|
||||
+}
|
||||
+
|
||||
+static int qca808x_led_blink_set(struct phy_device *phydev, u8 index,
|
||||
+ unsigned long *delay_on,
|
||||
+ unsigned long *delay_off)
|
||||
+{
|
||||
+ int ret;
|
||||
+ u16 reg;
|
||||
+
|
||||
+ if (index > 2)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ reg = QCA808X_MMD7_LED_FORCE_CTRL(index);
|
||||
+
|
||||
+ /* Set blink to 50% off, 50% on at 4Hz by default */
|
||||
+ ret = phy_modify_mmd(phydev, MDIO_MMD_AN, QCA808X_MMD7_LED_GLOBAL,
|
||||
+ QCA808X_LED_BLINK_FREQ_MASK | QCA808X_LED_BLINK_DUTY_MASK,
|
||||
+ QCA808X_LED_BLINK_FREQ_4HZ | QCA808X_LED_BLINK_DUTY_50_50);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ /* We use BLINK_1 for normal blinking */
|
||||
+ ret = phy_modify_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
+ QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_MODE_MASK,
|
||||
+ QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_BLINK_1);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ /* We set blink to 4Hz, aka 250ms */
|
||||
+ *delay_on = 250 / 2;
|
||||
+ *delay_off = 250 / 2;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca808x_led_polarity_set(struct phy_device *phydev, int index,
|
||||
+ unsigned long modes)
|
||||
+{
|
||||
+ struct at803x_priv *priv = phydev->priv;
|
||||
+ bool active_low = false;
|
||||
+ u32 mode;
|
||||
+
|
||||
+ for_each_set_bit(mode, &modes, __PHY_LED_MODES_NUM) {
|
||||
+ switch (mode) {
|
||||
+ case PHY_LED_ACTIVE_LOW:
|
||||
+ active_low = true;
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ /* PHY polarity is global and can't be set per LED.
|
||||
+ * To detect this, check if last requested polarity mode
|
||||
+ * match the new one.
|
||||
+ */
|
||||
+ if (priv->led_polarity_mode >= 0 &&
|
||||
+ priv->led_polarity_mode != active_low) {
|
||||
+ phydev_err(phydev, "PHY polarity is global. Mismatched polarity on different LED\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ /* Save the last PHY polarity mode */
|
||||
+ priv->led_polarity_mode = active_low;
|
||||
+
|
||||
+ return phy_modify_mmd(phydev, MDIO_MMD_AN,
|
||||
+ QCA808X_MMD7_LED_POLARITY_CTRL,
|
||||
+ QCA808X_LED_ACTIVE_HIGH,
|
||||
+ active_low ? 0 : QCA808X_LED_ACTIVE_HIGH);
|
||||
+}
|
||||
+
|
||||
static struct phy_driver at803x_driver[] = {
|
||||
{
|
||||
/* Qualcomm Atheros AR8035 */
|
||||
@@ -2411,6 +2732,12 @@ static struct phy_driver at803x_driver[]
|
||||
.cable_test_start = qca808x_cable_test_start,
|
||||
.cable_test_get_status = qca808x_cable_test_get_status,
|
||||
.link_change_notify = qca808x_link_change_notify,
|
||||
+ .led_brightness_set = qca808x_led_brightness_set,
|
||||
+ .led_blink_set = qca808x_led_blink_set,
|
||||
+ .led_hw_is_supported = qca808x_led_hw_is_supported,
|
||||
+ .led_hw_control_set = qca808x_led_hw_control_set,
|
||||
+ .led_hw_control_get = qca808x_led_hw_control_get,
|
||||
+ .led_polarity_set = qca808x_led_polarity_set,
|
||||
}, };
|
||||
|
||||
module_phy_driver(at803x_driver);
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,243 @@
|
||||
From 6fb760972c49490b03f3db2ad64cf30bdd28c54a Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Mon, 29 Jan 2024 15:15:20 +0100
|
||||
Subject: [PATCH 2/5] net: phy: qcom: create and move functions to shared
|
||||
library
|
||||
|
||||
Create and move functions to shared library in preparation for qca83xx
|
||||
PHY Family to be detached from at803x driver.
|
||||
|
||||
Only the shared defines are moved to the shared qcom.h header.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Link: https://lore.kernel.org/r/20240129141600.2592-3-ansuelsmth@gmail.com
|
||||
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
|
||||
---
|
||||
drivers/net/phy/qcom/Kconfig | 4 ++
|
||||
drivers/net/phy/qcom/Makefile | 1 +
|
||||
drivers/net/phy/qcom/at803x.c | 69 +----------------------------
|
||||
drivers/net/phy/qcom/qcom-phy-lib.c | 53 ++++++++++++++++++++++
|
||||
drivers/net/phy/qcom/qcom.h | 34 ++++++++++++++
|
||||
5 files changed, 94 insertions(+), 67 deletions(-)
|
||||
create mode 100644 drivers/net/phy/qcom/qcom-phy-lib.c
|
||||
create mode 100644 drivers/net/phy/qcom/qcom.h
|
||||
|
||||
--- a/drivers/net/phy/qcom/Kconfig
|
||||
+++ b/drivers/net/phy/qcom/Kconfig
|
||||
@@ -1,6 +1,10 @@
|
||||
# SPDX-License-Identifier: GPL-2.0-only
|
||||
+config QCOM_NET_PHYLIB
|
||||
+ tristate
|
||||
+
|
||||
config AT803X_PHY
|
||||
tristate "Qualcomm Atheros AR803X PHYs and QCA833x PHYs"
|
||||
+ select QCOM_NET_PHYLIB
|
||||
depends on REGULATOR
|
||||
help
|
||||
Currently supports the AR8030, AR8031, AR8033, AR8035 and internal
|
||||
--- a/drivers/net/phy/qcom/Makefile
|
||||
+++ b/drivers/net/phy/qcom/Makefile
|
||||
@@ -1,2 +1,3 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
+obj-$(CONFIG_QCOM_NET_PHYLIB) += qcom-phy-lib.o
|
||||
obj-$(CONFIG_AT803X_PHY) += at803x.o
|
||||
--- a/drivers/net/phy/qcom/at803x.c
|
||||
+++ b/drivers/net/phy/qcom/at803x.c
|
||||
@@ -22,6 +22,8 @@
|
||||
#include <linux/sfp.h>
|
||||
#include <dt-bindings/net/qca-ar803x.h>
|
||||
|
||||
+#include "qcom.h"
|
||||
+
|
||||
#define AT803X_SPECIFIC_FUNCTION_CONTROL 0x10
|
||||
#define AT803X_SFC_ASSERT_CRS BIT(11)
|
||||
#define AT803X_SFC_FORCE_LINK BIT(10)
|
||||
@@ -84,9 +86,6 @@
|
||||
#define AT803X_REG_CHIP_CONFIG 0x1f
|
||||
#define AT803X_BT_BX_REG_SEL 0x8000
|
||||
|
||||
-#define AT803X_DEBUG_ADDR 0x1D
|
||||
-#define AT803X_DEBUG_DATA 0x1E
|
||||
-
|
||||
#define AT803X_MODE_CFG_MASK 0x0F
|
||||
#define AT803X_MODE_CFG_BASET_RGMII 0x00
|
||||
#define AT803X_MODE_CFG_BASET_SGMII 0x01
|
||||
@@ -103,19 +102,6 @@
|
||||
#define AT803X_PSSR 0x11 /*PHY-Specific Status Register*/
|
||||
#define AT803X_PSSR_MR_AN_COMPLETE 0x0200
|
||||
|
||||
-#define AT803X_DEBUG_ANALOG_TEST_CTRL 0x00
|
||||
-#define QCA8327_DEBUG_MANU_CTRL_EN BIT(2)
|
||||
-#define QCA8337_DEBUG_MANU_CTRL_EN GENMASK(3, 2)
|
||||
-#define AT803X_DEBUG_RX_CLK_DLY_EN BIT(15)
|
||||
-
|
||||
-#define AT803X_DEBUG_SYSTEM_CTRL_MODE 0x05
|
||||
-#define AT803X_DEBUG_TX_CLK_DLY_EN BIT(8)
|
||||
-
|
||||
-#define AT803X_DEBUG_REG_HIB_CTRL 0x0b
|
||||
-#define AT803X_DEBUG_HIB_CTRL_SEL_RST_80U BIT(10)
|
||||
-#define AT803X_DEBUG_HIB_CTRL_EN_ANY_CHANGE BIT(13)
|
||||
-#define AT803X_DEBUG_HIB_CTRL_PS_HIB_EN BIT(15)
|
||||
-
|
||||
#define AT803X_DEBUG_REG_3C 0x3C
|
||||
|
||||
#define AT803X_DEBUG_REG_GREEN 0x3D
|
||||
@@ -393,18 +379,6 @@ MODULE_DESCRIPTION("Qualcomm Atheros AR8
|
||||
MODULE_AUTHOR("Matus Ujhelyi");
|
||||
MODULE_LICENSE("GPL");
|
||||
|
||||
-enum stat_access_type {
|
||||
- PHY,
|
||||
- MMD
|
||||
-};
|
||||
-
|
||||
-struct at803x_hw_stat {
|
||||
- const char *string;
|
||||
- u8 reg;
|
||||
- u32 mask;
|
||||
- enum stat_access_type access_type;
|
||||
-};
|
||||
-
|
||||
static struct at803x_hw_stat qca83xx_hw_stats[] = {
|
||||
{ "phy_idle_errors", 0xa, GENMASK(7, 0), PHY},
|
||||
{ "phy_receive_errors", 0x15, GENMASK(15, 0), PHY},
|
||||
@@ -439,45 +413,6 @@ struct at803x_context {
|
||||
u16 led_control;
|
||||
};
|
||||
|
||||
-static int at803x_debug_reg_write(struct phy_device *phydev, u16 reg, u16 data)
|
||||
-{
|
||||
- int ret;
|
||||
-
|
||||
- ret = phy_write(phydev, AT803X_DEBUG_ADDR, reg);
|
||||
- if (ret < 0)
|
||||
- return ret;
|
||||
-
|
||||
- return phy_write(phydev, AT803X_DEBUG_DATA, data);
|
||||
-}
|
||||
-
|
||||
-static int at803x_debug_reg_read(struct phy_device *phydev, u16 reg)
|
||||
-{
|
||||
- int ret;
|
||||
-
|
||||
- ret = phy_write(phydev, AT803X_DEBUG_ADDR, reg);
|
||||
- if (ret < 0)
|
||||
- return ret;
|
||||
-
|
||||
- return phy_read(phydev, AT803X_DEBUG_DATA);
|
||||
-}
|
||||
-
|
||||
-static int at803x_debug_reg_mask(struct phy_device *phydev, u16 reg,
|
||||
- u16 clear, u16 set)
|
||||
-{
|
||||
- u16 val;
|
||||
- int ret;
|
||||
-
|
||||
- ret = at803x_debug_reg_read(phydev, reg);
|
||||
- if (ret < 0)
|
||||
- return ret;
|
||||
-
|
||||
- val = ret & 0xffff;
|
||||
- val &= ~clear;
|
||||
- val |= set;
|
||||
-
|
||||
- return phy_write(phydev, AT803X_DEBUG_DATA, val);
|
||||
-}
|
||||
-
|
||||
static int at803x_write_page(struct phy_device *phydev, int page)
|
||||
{
|
||||
int mask;
|
||||
--- /dev/null
|
||||
+++ b/drivers/net/phy/qcom/qcom-phy-lib.c
|
||||
@@ -0,0 +1,53 @@
|
||||
+// SPDX-License-Identifier: GPL-2.0
|
||||
+
|
||||
+#include <linux/phy.h>
|
||||
+#include <linux/module.h>
|
||||
+
|
||||
+#include "qcom.h"
|
||||
+
|
||||
+MODULE_DESCRIPTION("Qualcomm PHY driver Common Functions");
|
||||
+MODULE_AUTHOR("Matus Ujhelyi");
|
||||
+MODULE_AUTHOR("Christian Marangi <ansuelsmth@gmail.com>");
|
||||
+MODULE_LICENSE("GPL");
|
||||
+
|
||||
+int at803x_debug_reg_read(struct phy_device *phydev, u16 reg)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = phy_write(phydev, AT803X_DEBUG_ADDR, reg);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+
|
||||
+ return phy_read(phydev, AT803X_DEBUG_DATA);
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(at803x_debug_reg_read);
|
||||
+
|
||||
+int at803x_debug_reg_mask(struct phy_device *phydev, u16 reg,
|
||||
+ u16 clear, u16 set)
|
||||
+{
|
||||
+ u16 val;
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = at803x_debug_reg_read(phydev, reg);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+
|
||||
+ val = ret & 0xffff;
|
||||
+ val &= ~clear;
|
||||
+ val |= set;
|
||||
+
|
||||
+ return phy_write(phydev, AT803X_DEBUG_DATA, val);
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(at803x_debug_reg_mask);
|
||||
+
|
||||
+int at803x_debug_reg_write(struct phy_device *phydev, u16 reg, u16 data)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = phy_write(phydev, AT803X_DEBUG_ADDR, reg);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+
|
||||
+ return phy_write(phydev, AT803X_DEBUG_DATA, data);
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(at803x_debug_reg_write);
|
||||
--- /dev/null
|
||||
+++ b/drivers/net/phy/qcom/qcom.h
|
||||
@@ -0,0 +1,34 @@
|
||||
+/* SPDX-License-Identifier: GPL-2.0 */
|
||||
+
|
||||
+#define AT803X_DEBUG_ADDR 0x1D
|
||||
+#define AT803X_DEBUG_DATA 0x1E
|
||||
+
|
||||
+#define AT803X_DEBUG_ANALOG_TEST_CTRL 0x00
|
||||
+#define QCA8327_DEBUG_MANU_CTRL_EN BIT(2)
|
||||
+#define QCA8337_DEBUG_MANU_CTRL_EN GENMASK(3, 2)
|
||||
+#define AT803X_DEBUG_RX_CLK_DLY_EN BIT(15)
|
||||
+
|
||||
+#define AT803X_DEBUG_SYSTEM_CTRL_MODE 0x05
|
||||
+#define AT803X_DEBUG_TX_CLK_DLY_EN BIT(8)
|
||||
+
|
||||
+#define AT803X_DEBUG_REG_HIB_CTRL 0x0b
|
||||
+#define AT803X_DEBUG_HIB_CTRL_SEL_RST_80U BIT(10)
|
||||
+#define AT803X_DEBUG_HIB_CTRL_EN_ANY_CHANGE BIT(13)
|
||||
+#define AT803X_DEBUG_HIB_CTRL_PS_HIB_EN BIT(15)
|
||||
+
|
||||
+enum stat_access_type {
|
||||
+ PHY,
|
||||
+ MMD
|
||||
+};
|
||||
+
|
||||
+struct at803x_hw_stat {
|
||||
+ const char *string;
|
||||
+ u8 reg;
|
||||
+ u32 mask;
|
||||
+ enum stat_access_type access_type;
|
||||
+};
|
||||
+
|
||||
+int at803x_debug_reg_read(struct phy_device *phydev, u16 reg);
|
||||
+int at803x_debug_reg_mask(struct phy_device *phydev, u16 reg,
|
||||
+ u16 clear, u16 set);
|
||||
+int at803x_debug_reg_write(struct phy_device *phydev, u16 reg, u16 data);
|
@ -0,0 +1,638 @@
|
||||
From 2e45d404d99d43bb7127b74b5dea8818df64996c Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Mon, 29 Jan 2024 15:15:21 +0100
|
||||
Subject: [PATCH 3/5] net: phy: qcom: deatch qca83xx PHY driver from at803x
|
||||
|
||||
Deatch qca83xx PHY driver from at803x.
|
||||
|
||||
The QCA83xx PHYs implement specific function and doesn't use generic
|
||||
at803x so it can be detached from the driver and moved to a dedicated
|
||||
one.
|
||||
|
||||
Probe function and priv struct is reimplemented to allocate and use
|
||||
only the qca83xx specific data. Unused data from at803x PHY driver
|
||||
are dropped from at803x priv struct.
|
||||
|
||||
This is to make slimmer PHY drivers instead of including lots of bloat
|
||||
that would never be used in specific SoC.
|
||||
|
||||
A new Kconfig flag QCA83XX_PHY is introduced to compile the new
|
||||
introduced PHY driver.
|
||||
|
||||
As the Kconfig name starts with Qualcomm the same order is kept.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Link: https://lore.kernel.org/r/20240129141600.2592-4-ansuelsmth@gmail.com
|
||||
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
|
||||
---
|
||||
drivers/net/phy/qcom/Kconfig | 11 +-
|
||||
drivers/net/phy/qcom/Makefile | 1 +
|
||||
drivers/net/phy/qcom/at803x.c | 235 ----------------------------
|
||||
drivers/net/phy/qcom/qca83xx.c | 275 +++++++++++++++++++++++++++++++++
|
||||
4 files changed, 284 insertions(+), 238 deletions(-)
|
||||
create mode 100644 drivers/net/phy/qcom/qca83xx.c
|
||||
|
||||
--- a/drivers/net/phy/qcom/Kconfig
|
||||
+++ b/drivers/net/phy/qcom/Kconfig
|
||||
@@ -3,9 +3,14 @@ config QCOM_NET_PHYLIB
|
||||
tristate
|
||||
|
||||
config AT803X_PHY
|
||||
- tristate "Qualcomm Atheros AR803X PHYs and QCA833x PHYs"
|
||||
+ tristate "Qualcomm Atheros AR803X PHYs"
|
||||
select QCOM_NET_PHYLIB
|
||||
depends on REGULATOR
|
||||
help
|
||||
- Currently supports the AR8030, AR8031, AR8033, AR8035 and internal
|
||||
- QCA8337(Internal qca8k PHY) model
|
||||
+ Currently supports the AR8030, AR8031, AR8033, AR8035 model
|
||||
+
|
||||
+config QCA83XX_PHY
|
||||
+ tristate "Qualcomm Atheros QCA833x PHYs"
|
||||
+ select QCOM_NET_PHYLIB
|
||||
+ help
|
||||
+ Currently supports the internal QCA8337(Internal qca8k PHY) model
|
||||
--- a/drivers/net/phy/qcom/Makefile
|
||||
+++ b/drivers/net/phy/qcom/Makefile
|
||||
@@ -1,3 +1,4 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
obj-$(CONFIG_QCOM_NET_PHYLIB) += qcom-phy-lib.o
|
||||
obj-$(CONFIG_AT803X_PHY) += at803x.o
|
||||
+obj-$(CONFIG_QCA83XX_PHY) += qca83xx.o
|
||||
--- a/drivers/net/phy/qcom/at803x.c
|
||||
+++ b/drivers/net/phy/qcom/at803x.c
|
||||
@@ -102,17 +102,10 @@
|
||||
#define AT803X_PSSR 0x11 /*PHY-Specific Status Register*/
|
||||
#define AT803X_PSSR_MR_AN_COMPLETE 0x0200
|
||||
|
||||
-#define AT803X_DEBUG_REG_3C 0x3C
|
||||
-
|
||||
-#define AT803X_DEBUG_REG_GREEN 0x3D
|
||||
-#define AT803X_DEBUG_GATE_CLK_IN1000 BIT(6)
|
||||
-
|
||||
#define AT803X_DEBUG_REG_1F 0x1F
|
||||
#define AT803X_DEBUG_PLL_ON BIT(2)
|
||||
#define AT803X_DEBUG_RGMII_1V8 BIT(3)
|
||||
|
||||
-#define MDIO_AZ_DEBUG 0x800D
|
||||
-
|
||||
/* AT803x supports either the XTAL input pad, an internal PLL or the
|
||||
* DSP as clock reference for the clock output pad. The XTAL reference
|
||||
* is only used for 25 MHz output, all other frequencies need the PLL.
|
||||
@@ -163,13 +156,7 @@
|
||||
|
||||
#define QCA8081_PHY_ID 0x004dd101
|
||||
|
||||
-#define QCA8327_A_PHY_ID 0x004dd033
|
||||
-#define QCA8327_B_PHY_ID 0x004dd034
|
||||
-#define QCA8337_PHY_ID 0x004dd036
|
||||
#define QCA9561_PHY_ID 0x004dd042
|
||||
-#define QCA8K_PHY_ID_MASK 0xffffffff
|
||||
-
|
||||
-#define QCA8K_DEVFLAGS_REVISION_MASK GENMASK(2, 0)
|
||||
|
||||
#define AT803X_PAGE_FIBER 0
|
||||
#define AT803X_PAGE_COPPER 1
|
||||
@@ -379,12 +366,6 @@ MODULE_DESCRIPTION("Qualcomm Atheros AR8
|
||||
MODULE_AUTHOR("Matus Ujhelyi");
|
||||
MODULE_LICENSE("GPL");
|
||||
|
||||
-static struct at803x_hw_stat qca83xx_hw_stats[] = {
|
||||
- { "phy_idle_errors", 0xa, GENMASK(7, 0), PHY},
|
||||
- { "phy_receive_errors", 0x15, GENMASK(15, 0), PHY},
|
||||
- { "eee_wake_errors", 0x16, GENMASK(15, 0), MMD},
|
||||
-};
|
||||
-
|
||||
struct at803x_ss_mask {
|
||||
u16 speed_mask;
|
||||
u8 speed_shift;
|
||||
@@ -400,7 +381,6 @@ struct at803x_priv {
|
||||
bool is_1000basex;
|
||||
struct regulator_dev *vddio_rdev;
|
||||
struct regulator_dev *vddh_rdev;
|
||||
- u64 stats[ARRAY_SIZE(qca83xx_hw_stats)];
|
||||
int led_polarity_mode;
|
||||
};
|
||||
|
||||
@@ -564,53 +544,6 @@ static void at803x_get_wol(struct phy_de
|
||||
wol->wolopts |= WAKE_MAGIC;
|
||||
}
|
||||
|
||||
-static int qca83xx_get_sset_count(struct phy_device *phydev)
|
||||
-{
|
||||
- return ARRAY_SIZE(qca83xx_hw_stats);
|
||||
-}
|
||||
-
|
||||
-static void qca83xx_get_strings(struct phy_device *phydev, u8 *data)
|
||||
-{
|
||||
- int i;
|
||||
-
|
||||
- for (i = 0; i < ARRAY_SIZE(qca83xx_hw_stats); i++) {
|
||||
- strscpy(data + i * ETH_GSTRING_LEN,
|
||||
- qca83xx_hw_stats[i].string, ETH_GSTRING_LEN);
|
||||
- }
|
||||
-}
|
||||
-
|
||||
-static u64 qca83xx_get_stat(struct phy_device *phydev, int i)
|
||||
-{
|
||||
- struct at803x_hw_stat stat = qca83xx_hw_stats[i];
|
||||
- struct at803x_priv *priv = phydev->priv;
|
||||
- int val;
|
||||
- u64 ret;
|
||||
-
|
||||
- if (stat.access_type == MMD)
|
||||
- val = phy_read_mmd(phydev, MDIO_MMD_PCS, stat.reg);
|
||||
- else
|
||||
- val = phy_read(phydev, stat.reg);
|
||||
-
|
||||
- if (val < 0) {
|
||||
- ret = U64_MAX;
|
||||
- } else {
|
||||
- val = val & stat.mask;
|
||||
- priv->stats[i] += val;
|
||||
- ret = priv->stats[i];
|
||||
- }
|
||||
-
|
||||
- return ret;
|
||||
-}
|
||||
-
|
||||
-static void qca83xx_get_stats(struct phy_device *phydev,
|
||||
- struct ethtool_stats *stats, u64 *data)
|
||||
-{
|
||||
- int i;
|
||||
-
|
||||
- for (i = 0; i < ARRAY_SIZE(qca83xx_hw_stats); i++)
|
||||
- data[i] = qca83xx_get_stat(phydev, i);
|
||||
-}
|
||||
-
|
||||
static int at803x_suspend(struct phy_device *phydev)
|
||||
{
|
||||
int value;
|
||||
@@ -1707,124 +1640,6 @@ static int at8035_probe(struct phy_devic
|
||||
return at8035_parse_dt(phydev);
|
||||
}
|
||||
|
||||
-static int qca83xx_config_init(struct phy_device *phydev)
|
||||
-{
|
||||
- u8 switch_revision;
|
||||
-
|
||||
- switch_revision = phydev->dev_flags & QCA8K_DEVFLAGS_REVISION_MASK;
|
||||
-
|
||||
- switch (switch_revision) {
|
||||
- case 1:
|
||||
- /* For 100M waveform */
|
||||
- at803x_debug_reg_write(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL, 0x02ea);
|
||||
- /* Turn on Gigabit clock */
|
||||
- at803x_debug_reg_write(phydev, AT803X_DEBUG_REG_GREEN, 0x68a0);
|
||||
- break;
|
||||
-
|
||||
- case 2:
|
||||
- phy_write_mmd(phydev, MDIO_MMD_AN, MDIO_AN_EEE_ADV, 0x0);
|
||||
- fallthrough;
|
||||
- case 4:
|
||||
- phy_write_mmd(phydev, MDIO_MMD_PCS, MDIO_AZ_DEBUG, 0x803f);
|
||||
- at803x_debug_reg_write(phydev, AT803X_DEBUG_REG_GREEN, 0x6860);
|
||||
- at803x_debug_reg_write(phydev, AT803X_DEBUG_SYSTEM_CTRL_MODE, 0x2c46);
|
||||
- at803x_debug_reg_write(phydev, AT803X_DEBUG_REG_3C, 0x6000);
|
||||
- break;
|
||||
- }
|
||||
-
|
||||
- /* Following original QCA sourcecode set port to prefer master */
|
||||
- phy_set_bits(phydev, MII_CTRL1000, CTL1000_PREFER_MASTER);
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
-
|
||||
-static int qca8327_config_init(struct phy_device *phydev)
|
||||
-{
|
||||
- /* QCA8327 require DAC amplitude adjustment for 100m set to +6%.
|
||||
- * Disable on init and enable only with 100m speed following
|
||||
- * qca original source code.
|
||||
- */
|
||||
- at803x_debug_reg_mask(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL,
|
||||
- QCA8327_DEBUG_MANU_CTRL_EN, 0);
|
||||
-
|
||||
- return qca83xx_config_init(phydev);
|
||||
-}
|
||||
-
|
||||
-static void qca83xx_link_change_notify(struct phy_device *phydev)
|
||||
-{
|
||||
- /* Set DAC Amplitude adjustment to +6% for 100m on link running */
|
||||
- if (phydev->state == PHY_RUNNING) {
|
||||
- if (phydev->speed == SPEED_100)
|
||||
- at803x_debug_reg_mask(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL,
|
||||
- QCA8327_DEBUG_MANU_CTRL_EN,
|
||||
- QCA8327_DEBUG_MANU_CTRL_EN);
|
||||
- } else {
|
||||
- /* Reset DAC Amplitude adjustment */
|
||||
- at803x_debug_reg_mask(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL,
|
||||
- QCA8327_DEBUG_MANU_CTRL_EN, 0);
|
||||
- }
|
||||
-}
|
||||
-
|
||||
-static int qca83xx_resume(struct phy_device *phydev)
|
||||
-{
|
||||
- int ret, val;
|
||||
-
|
||||
- /* Skip reset if not suspended */
|
||||
- if (!phydev->suspended)
|
||||
- return 0;
|
||||
-
|
||||
- /* Reinit the port, reset values set by suspend */
|
||||
- qca83xx_config_init(phydev);
|
||||
-
|
||||
- /* Reset the port on port resume */
|
||||
- phy_set_bits(phydev, MII_BMCR, BMCR_RESET | BMCR_ANENABLE);
|
||||
-
|
||||
- /* On resume from suspend the switch execute a reset and
|
||||
- * restart auto-negotiation. Wait for reset to complete.
|
||||
- */
|
||||
- ret = phy_read_poll_timeout(phydev, MII_BMCR, val, !(val & BMCR_RESET),
|
||||
- 50000, 600000, true);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- usleep_range(1000, 2000);
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
-
|
||||
-static int qca83xx_suspend(struct phy_device *phydev)
|
||||
-{
|
||||
- at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_GREEN,
|
||||
- AT803X_DEBUG_GATE_CLK_IN1000, 0);
|
||||
-
|
||||
- at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_HIB_CTRL,
|
||||
- AT803X_DEBUG_HIB_CTRL_EN_ANY_CHANGE |
|
||||
- AT803X_DEBUG_HIB_CTRL_SEL_RST_80U, 0);
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
-
|
||||
-static int qca8337_suspend(struct phy_device *phydev)
|
||||
-{
|
||||
- /* Only QCA8337 support actual suspend. */
|
||||
- genphy_suspend(phydev);
|
||||
-
|
||||
- return qca83xx_suspend(phydev);
|
||||
-}
|
||||
-
|
||||
-static int qca8327_suspend(struct phy_device *phydev)
|
||||
-{
|
||||
- u16 mask = 0;
|
||||
-
|
||||
- /* QCA8327 cause port unreliability when phy suspend
|
||||
- * is set.
|
||||
- */
|
||||
- mask |= ~(BMCR_SPEED1000 | BMCR_FULLDPLX);
|
||||
- phy_modify(phydev, MII_BMCR, mask, 0);
|
||||
-
|
||||
- return qca83xx_suspend(phydev);
|
||||
-}
|
||||
-
|
||||
static int qca808x_phy_fast_retrain_config(struct phy_device *phydev)
|
||||
{
|
||||
int ret;
|
||||
@@ -2599,53 +2414,6 @@ static struct phy_driver at803x_driver[]
|
||||
.soft_reset = genphy_soft_reset,
|
||||
.config_aneg = at803x_config_aneg,
|
||||
}, {
|
||||
- /* QCA8337 */
|
||||
- .phy_id = QCA8337_PHY_ID,
|
||||
- .phy_id_mask = QCA8K_PHY_ID_MASK,
|
||||
- .name = "Qualcomm Atheros 8337 internal PHY",
|
||||
- /* PHY_GBIT_FEATURES */
|
||||
- .probe = at803x_probe,
|
||||
- .flags = PHY_IS_INTERNAL,
|
||||
- .config_init = qca83xx_config_init,
|
||||
- .soft_reset = genphy_soft_reset,
|
||||
- .get_sset_count = qca83xx_get_sset_count,
|
||||
- .get_strings = qca83xx_get_strings,
|
||||
- .get_stats = qca83xx_get_stats,
|
||||
- .suspend = qca8337_suspend,
|
||||
- .resume = qca83xx_resume,
|
||||
-}, {
|
||||
- /* QCA8327-A from switch QCA8327-AL1A */
|
||||
- .phy_id = QCA8327_A_PHY_ID,
|
||||
- .phy_id_mask = QCA8K_PHY_ID_MASK,
|
||||
- .name = "Qualcomm Atheros 8327-A internal PHY",
|
||||
- /* PHY_GBIT_FEATURES */
|
||||
- .link_change_notify = qca83xx_link_change_notify,
|
||||
- .probe = at803x_probe,
|
||||
- .flags = PHY_IS_INTERNAL,
|
||||
- .config_init = qca8327_config_init,
|
||||
- .soft_reset = genphy_soft_reset,
|
||||
- .get_sset_count = qca83xx_get_sset_count,
|
||||
- .get_strings = qca83xx_get_strings,
|
||||
- .get_stats = qca83xx_get_stats,
|
||||
- .suspend = qca8327_suspend,
|
||||
- .resume = qca83xx_resume,
|
||||
-}, {
|
||||
- /* QCA8327-B from switch QCA8327-BL1A */
|
||||
- .phy_id = QCA8327_B_PHY_ID,
|
||||
- .phy_id_mask = QCA8K_PHY_ID_MASK,
|
||||
- .name = "Qualcomm Atheros 8327-B internal PHY",
|
||||
- /* PHY_GBIT_FEATURES */
|
||||
- .link_change_notify = qca83xx_link_change_notify,
|
||||
- .probe = at803x_probe,
|
||||
- .flags = PHY_IS_INTERNAL,
|
||||
- .config_init = qca8327_config_init,
|
||||
- .soft_reset = genphy_soft_reset,
|
||||
- .get_sset_count = qca83xx_get_sset_count,
|
||||
- .get_strings = qca83xx_get_strings,
|
||||
- .get_stats = qca83xx_get_stats,
|
||||
- .suspend = qca8327_suspend,
|
||||
- .resume = qca83xx_resume,
|
||||
-}, {
|
||||
/* Qualcomm QCA8081 */
|
||||
PHY_ID_MATCH_EXACT(QCA8081_PHY_ID),
|
||||
.name = "Qualcomm QCA8081",
|
||||
@@ -2683,9 +2451,6 @@ static struct mdio_device_id __maybe_unu
|
||||
{ PHY_ID_MATCH_EXACT(ATH8032_PHY_ID) },
|
||||
{ PHY_ID_MATCH_EXACT(ATH8035_PHY_ID) },
|
||||
{ PHY_ID_MATCH_EXACT(ATH9331_PHY_ID) },
|
||||
- { PHY_ID_MATCH_EXACT(QCA8337_PHY_ID) },
|
||||
- { PHY_ID_MATCH_EXACT(QCA8327_A_PHY_ID) },
|
||||
- { PHY_ID_MATCH_EXACT(QCA8327_B_PHY_ID) },
|
||||
{ PHY_ID_MATCH_EXACT(QCA9561_PHY_ID) },
|
||||
{ PHY_ID_MATCH_EXACT(QCA8081_PHY_ID) },
|
||||
{ }
|
||||
--- /dev/null
|
||||
+++ b/drivers/net/phy/qcom/qca83xx.c
|
||||
@@ -0,0 +1,275 @@
|
||||
+// SPDX-License-Identifier: GPL-2.0+
|
||||
+
|
||||
+#include <linux/phy.h>
|
||||
+#include <linux/module.h>
|
||||
+
|
||||
+#include "qcom.h"
|
||||
+
|
||||
+#define AT803X_DEBUG_REG_3C 0x3C
|
||||
+
|
||||
+#define AT803X_DEBUG_REG_GREEN 0x3D
|
||||
+#define AT803X_DEBUG_GATE_CLK_IN1000 BIT(6)
|
||||
+
|
||||
+#define MDIO_AZ_DEBUG 0x800D
|
||||
+
|
||||
+#define QCA8327_A_PHY_ID 0x004dd033
|
||||
+#define QCA8327_B_PHY_ID 0x004dd034
|
||||
+#define QCA8337_PHY_ID 0x004dd036
|
||||
+#define QCA8K_PHY_ID_MASK 0xffffffff
|
||||
+
|
||||
+#define QCA8K_DEVFLAGS_REVISION_MASK GENMASK(2, 0)
|
||||
+
|
||||
+static struct at803x_hw_stat qca83xx_hw_stats[] = {
|
||||
+ { "phy_idle_errors", 0xa, GENMASK(7, 0), PHY},
|
||||
+ { "phy_receive_errors", 0x15, GENMASK(15, 0), PHY},
|
||||
+ { "eee_wake_errors", 0x16, GENMASK(15, 0), MMD},
|
||||
+};
|
||||
+
|
||||
+struct qca83xx_priv {
|
||||
+ u64 stats[ARRAY_SIZE(qca83xx_hw_stats)];
|
||||
+};
|
||||
+
|
||||
+MODULE_DESCRIPTION("Qualcomm Atheros QCA83XX PHY driver");
|
||||
+MODULE_AUTHOR("Matus Ujhelyi");
|
||||
+MODULE_AUTHOR("Christian Marangi <ansuelsmth@gmail.com>");
|
||||
+MODULE_LICENSE("GPL");
|
||||
+
|
||||
+static int qca83xx_get_sset_count(struct phy_device *phydev)
|
||||
+{
|
||||
+ return ARRAY_SIZE(qca83xx_hw_stats);
|
||||
+}
|
||||
+
|
||||
+static void qca83xx_get_strings(struct phy_device *phydev, u8 *data)
|
||||
+{
|
||||
+ int i;
|
||||
+
|
||||
+ for (i = 0; i < ARRAY_SIZE(qca83xx_hw_stats); i++) {
|
||||
+ strscpy(data + i * ETH_GSTRING_LEN,
|
||||
+ qca83xx_hw_stats[i].string, ETH_GSTRING_LEN);
|
||||
+ }
|
||||
+}
|
||||
+
|
||||
+static u64 qca83xx_get_stat(struct phy_device *phydev, int i)
|
||||
+{
|
||||
+ struct at803x_hw_stat stat = qca83xx_hw_stats[i];
|
||||
+ struct qca83xx_priv *priv = phydev->priv;
|
||||
+ int val;
|
||||
+ u64 ret;
|
||||
+
|
||||
+ if (stat.access_type == MMD)
|
||||
+ val = phy_read_mmd(phydev, MDIO_MMD_PCS, stat.reg);
|
||||
+ else
|
||||
+ val = phy_read(phydev, stat.reg);
|
||||
+
|
||||
+ if (val < 0) {
|
||||
+ ret = U64_MAX;
|
||||
+ } else {
|
||||
+ val = val & stat.mask;
|
||||
+ priv->stats[i] += val;
|
||||
+ ret = priv->stats[i];
|
||||
+ }
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
+static void qca83xx_get_stats(struct phy_device *phydev,
|
||||
+ struct ethtool_stats *stats, u64 *data)
|
||||
+{
|
||||
+ int i;
|
||||
+
|
||||
+ for (i = 0; i < ARRAY_SIZE(qca83xx_hw_stats); i++)
|
||||
+ data[i] = qca83xx_get_stat(phydev, i);
|
||||
+}
|
||||
+
|
||||
+static int qca83xx_probe(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct device *dev = &phydev->mdio.dev;
|
||||
+ struct qca83xx_priv *priv;
|
||||
+
|
||||
+ priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
|
||||
+ if (!priv)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ phydev->priv = priv;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca83xx_config_init(struct phy_device *phydev)
|
||||
+{
|
||||
+ u8 switch_revision;
|
||||
+
|
||||
+ switch_revision = phydev->dev_flags & QCA8K_DEVFLAGS_REVISION_MASK;
|
||||
+
|
||||
+ switch (switch_revision) {
|
||||
+ case 1:
|
||||
+ /* For 100M waveform */
|
||||
+ at803x_debug_reg_write(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL, 0x02ea);
|
||||
+ /* Turn on Gigabit clock */
|
||||
+ at803x_debug_reg_write(phydev, AT803X_DEBUG_REG_GREEN, 0x68a0);
|
||||
+ break;
|
||||
+
|
||||
+ case 2:
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_AN, MDIO_AN_EEE_ADV, 0x0);
|
||||
+ fallthrough;
|
||||
+ case 4:
|
||||
+ phy_write_mmd(phydev, MDIO_MMD_PCS, MDIO_AZ_DEBUG, 0x803f);
|
||||
+ at803x_debug_reg_write(phydev, AT803X_DEBUG_REG_GREEN, 0x6860);
|
||||
+ at803x_debug_reg_write(phydev, AT803X_DEBUG_SYSTEM_CTRL_MODE, 0x2c46);
|
||||
+ at803x_debug_reg_write(phydev, AT803X_DEBUG_REG_3C, 0x6000);
|
||||
+ break;
|
||||
+ }
|
||||
+
|
||||
+ /* Following original QCA sourcecode set port to prefer master */
|
||||
+ phy_set_bits(phydev, MII_CTRL1000, CTL1000_PREFER_MASTER);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca8327_config_init(struct phy_device *phydev)
|
||||
+{
|
||||
+ /* QCA8327 require DAC amplitude adjustment for 100m set to +6%.
|
||||
+ * Disable on init and enable only with 100m speed following
|
||||
+ * qca original source code.
|
||||
+ */
|
||||
+ at803x_debug_reg_mask(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL,
|
||||
+ QCA8327_DEBUG_MANU_CTRL_EN, 0);
|
||||
+
|
||||
+ return qca83xx_config_init(phydev);
|
||||
+}
|
||||
+
|
||||
+static void qca83xx_link_change_notify(struct phy_device *phydev)
|
||||
+{
|
||||
+ /* Set DAC Amplitude adjustment to +6% for 100m on link running */
|
||||
+ if (phydev->state == PHY_RUNNING) {
|
||||
+ if (phydev->speed == SPEED_100)
|
||||
+ at803x_debug_reg_mask(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL,
|
||||
+ QCA8327_DEBUG_MANU_CTRL_EN,
|
||||
+ QCA8327_DEBUG_MANU_CTRL_EN);
|
||||
+ } else {
|
||||
+ /* Reset DAC Amplitude adjustment */
|
||||
+ at803x_debug_reg_mask(phydev, AT803X_DEBUG_ANALOG_TEST_CTRL,
|
||||
+ QCA8327_DEBUG_MANU_CTRL_EN, 0);
|
||||
+ }
|
||||
+}
|
||||
+
|
||||
+static int qca83xx_resume(struct phy_device *phydev)
|
||||
+{
|
||||
+ int ret, val;
|
||||
+
|
||||
+ /* Skip reset if not suspended */
|
||||
+ if (!phydev->suspended)
|
||||
+ return 0;
|
||||
+
|
||||
+ /* Reinit the port, reset values set by suspend */
|
||||
+ qca83xx_config_init(phydev);
|
||||
+
|
||||
+ /* Reset the port on port resume */
|
||||
+ phy_set_bits(phydev, MII_BMCR, BMCR_RESET | BMCR_ANENABLE);
|
||||
+
|
||||
+ /* On resume from suspend the switch execute a reset and
|
||||
+ * restart auto-negotiation. Wait for reset to complete.
|
||||
+ */
|
||||
+ ret = phy_read_poll_timeout(phydev, MII_BMCR, val, !(val & BMCR_RESET),
|
||||
+ 50000, 600000, true);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ usleep_range(1000, 2000);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca83xx_suspend(struct phy_device *phydev)
|
||||
+{
|
||||
+ at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_GREEN,
|
||||
+ AT803X_DEBUG_GATE_CLK_IN1000, 0);
|
||||
+
|
||||
+ at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_HIB_CTRL,
|
||||
+ AT803X_DEBUG_HIB_CTRL_EN_ANY_CHANGE |
|
||||
+ AT803X_DEBUG_HIB_CTRL_SEL_RST_80U, 0);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca8337_suspend(struct phy_device *phydev)
|
||||
+{
|
||||
+ /* Only QCA8337 support actual suspend. */
|
||||
+ genphy_suspend(phydev);
|
||||
+
|
||||
+ return qca83xx_suspend(phydev);
|
||||
+}
|
||||
+
|
||||
+static int qca8327_suspend(struct phy_device *phydev)
|
||||
+{
|
||||
+ u16 mask = 0;
|
||||
+
|
||||
+ /* QCA8327 cause port unreliability when phy suspend
|
||||
+ * is set.
|
||||
+ */
|
||||
+ mask |= ~(BMCR_SPEED1000 | BMCR_FULLDPLX);
|
||||
+ phy_modify(phydev, MII_BMCR, mask, 0);
|
||||
+
|
||||
+ return qca83xx_suspend(phydev);
|
||||
+}
|
||||
+
|
||||
+static struct phy_driver qca83xx_driver[] = {
|
||||
+{
|
||||
+ /* QCA8337 */
|
||||
+ .phy_id = QCA8337_PHY_ID,
|
||||
+ .phy_id_mask = QCA8K_PHY_ID_MASK,
|
||||
+ .name = "Qualcomm Atheros 8337 internal PHY",
|
||||
+ /* PHY_GBIT_FEATURES */
|
||||
+ .probe = qca83xx_probe,
|
||||
+ .flags = PHY_IS_INTERNAL,
|
||||
+ .config_init = qca83xx_config_init,
|
||||
+ .soft_reset = genphy_soft_reset,
|
||||
+ .get_sset_count = qca83xx_get_sset_count,
|
||||
+ .get_strings = qca83xx_get_strings,
|
||||
+ .get_stats = qca83xx_get_stats,
|
||||
+ .suspend = qca8337_suspend,
|
||||
+ .resume = qca83xx_resume,
|
||||
+}, {
|
||||
+ /* QCA8327-A from switch QCA8327-AL1A */
|
||||
+ .phy_id = QCA8327_A_PHY_ID,
|
||||
+ .phy_id_mask = QCA8K_PHY_ID_MASK,
|
||||
+ .name = "Qualcomm Atheros 8327-A internal PHY",
|
||||
+ /* PHY_GBIT_FEATURES */
|
||||
+ .link_change_notify = qca83xx_link_change_notify,
|
||||
+ .probe = qca83xx_probe,
|
||||
+ .flags = PHY_IS_INTERNAL,
|
||||
+ .config_init = qca8327_config_init,
|
||||
+ .soft_reset = genphy_soft_reset,
|
||||
+ .get_sset_count = qca83xx_get_sset_count,
|
||||
+ .get_strings = qca83xx_get_strings,
|
||||
+ .get_stats = qca83xx_get_stats,
|
||||
+ .suspend = qca8327_suspend,
|
||||
+ .resume = qca83xx_resume,
|
||||
+}, {
|
||||
+ /* QCA8327-B from switch QCA8327-BL1A */
|
||||
+ .phy_id = QCA8327_B_PHY_ID,
|
||||
+ .phy_id_mask = QCA8K_PHY_ID_MASK,
|
||||
+ .name = "Qualcomm Atheros 8327-B internal PHY",
|
||||
+ /* PHY_GBIT_FEATURES */
|
||||
+ .link_change_notify = qca83xx_link_change_notify,
|
||||
+ .probe = qca83xx_probe,
|
||||
+ .flags = PHY_IS_INTERNAL,
|
||||
+ .config_init = qca8327_config_init,
|
||||
+ .soft_reset = genphy_soft_reset,
|
||||
+ .get_sset_count = qca83xx_get_sset_count,
|
||||
+ .get_strings = qca83xx_get_strings,
|
||||
+ .get_stats = qca83xx_get_stats,
|
||||
+ .suspend = qca8327_suspend,
|
||||
+ .resume = qca83xx_resume,
|
||||
+}, };
|
||||
+
|
||||
+module_phy_driver(qca83xx_driver);
|
||||
+
|
||||
+static struct mdio_device_id __maybe_unused qca83xx_tbl[] = {
|
||||
+ { PHY_ID_MATCH_EXACT(QCA8337_PHY_ID) },
|
||||
+ { PHY_ID_MATCH_EXACT(QCA8327_A_PHY_ID) },
|
||||
+ { PHY_ID_MATCH_EXACT(QCA8327_B_PHY_ID) },
|
||||
+ { }
|
||||
+};
|
||||
+
|
||||
+MODULE_DEVICE_TABLE(mdio, qca83xx_tbl);
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,28 @@
|
||||
From ebb30ccbbdbd6fae5177b676da4f4ac92bb4f635 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 15 Dec 2023 14:15:31 +0100
|
||||
Subject: [PATCH 1/4] net: phy: make addr type u8 in phy_package_shared struct
|
||||
|
||||
Switch addr type in phy_package_shared struct to u8.
|
||||
|
||||
The value is already checked to be non negative and to be less than
|
||||
PHY_MAX_ADDR, hence u8 is better suited than using int.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Russell King (Oracle) <rmk+kernel@armlinux.org.uk>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
include/linux/phy.h | 2 +-
|
||||
1 file changed, 1 insertion(+), 1 deletion(-)
|
||||
|
||||
--- a/include/linux/phy.h
|
||||
+++ b/include/linux/phy.h
|
||||
@@ -338,7 +338,7 @@ struct mdio_bus_stats {
|
||||
* phy_package_leave().
|
||||
*/
|
||||
struct phy_package_shared {
|
||||
- int addr;
|
||||
+ u8 addr;
|
||||
refcount_t refcnt;
|
||||
unsigned long flags;
|
||||
size_t priv_size;
|
@ -0,0 +1,341 @@
|
||||
From 9eea577eb1155fe4a183bc5e7bf269b0b2e7a6ba Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 15 Dec 2023 14:15:32 +0100
|
||||
Subject: [PATCH 2/4] net: phy: extend PHY package API to support multiple
|
||||
global address
|
||||
|
||||
Current API for PHY package are limited to single address to configure
|
||||
global settings for the PHY package.
|
||||
|
||||
It was found that some PHY package (for example the qca807x, a PHY
|
||||
package that is shipped with a bundle of 5 PHY) requires multiple PHY
|
||||
address to configure global settings. An example scenario is a PHY that
|
||||
have a dedicated PHY for PSGMII/serdes calibrarion and have a specific
|
||||
PHY in the package where the global PHY mode is set and affects every
|
||||
other PHY in the package.
|
||||
|
||||
Change the API in the following way:
|
||||
- Change phy_package_join() to take the base addr of the PHY package
|
||||
instead of the global PHY addr.
|
||||
- Make __/phy_package_write/read() require an additional arg that
|
||||
select what global PHY address to use by passing the offset from the
|
||||
base addr passed on phy_package_join().
|
||||
|
||||
Each user of this API is updated to follow this new implementation
|
||||
following a pattern where an enum is defined to declare the offset of the
|
||||
addr.
|
||||
|
||||
We also drop the check if shared is defined as any user of the
|
||||
phy_package_read/write is expected to use phy_package_join first. Misuse
|
||||
of this will correctly trigger a kernel panic for NULL pointer
|
||||
exception.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/bcm54140.c | 16 ++++++--
|
||||
drivers/net/phy/mscc/mscc.h | 5 +++
|
||||
drivers/net/phy/mscc/mscc_main.c | 4 +-
|
||||
drivers/net/phy/phy_device.c | 35 +++++++++--------
|
||||
include/linux/phy.h | 64 +++++++++++++++++++++-----------
|
||||
5 files changed, 80 insertions(+), 44 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/bcm54140.c
|
||||
+++ b/drivers/net/phy/bcm54140.c
|
||||
@@ -128,6 +128,10 @@
|
||||
#define BCM54140_DEFAULT_DOWNSHIFT 5
|
||||
#define BCM54140_MAX_DOWNSHIFT 9
|
||||
|
||||
+enum bcm54140_global_phy {
|
||||
+ BCM54140_BASE_ADDR = 0,
|
||||
+};
|
||||
+
|
||||
struct bcm54140_priv {
|
||||
int port;
|
||||
int base_addr;
|
||||
@@ -429,11 +433,13 @@ static int bcm54140_base_read_rdb(struct
|
||||
int ret;
|
||||
|
||||
phy_lock_mdio_bus(phydev);
|
||||
- ret = __phy_package_write(phydev, MII_BCM54XX_RDB_ADDR, rdb);
|
||||
+ ret = __phy_package_write(phydev, BCM54140_BASE_ADDR,
|
||||
+ MII_BCM54XX_RDB_ADDR, rdb);
|
||||
if (ret < 0)
|
||||
goto out;
|
||||
|
||||
- ret = __phy_package_read(phydev, MII_BCM54XX_RDB_DATA);
|
||||
+ ret = __phy_package_read(phydev, BCM54140_BASE_ADDR,
|
||||
+ MII_BCM54XX_RDB_DATA);
|
||||
|
||||
out:
|
||||
phy_unlock_mdio_bus(phydev);
|
||||
@@ -446,11 +452,13 @@ static int bcm54140_base_write_rdb(struc
|
||||
int ret;
|
||||
|
||||
phy_lock_mdio_bus(phydev);
|
||||
- ret = __phy_package_write(phydev, MII_BCM54XX_RDB_ADDR, rdb);
|
||||
+ ret = __phy_package_write(phydev, BCM54140_BASE_ADDR,
|
||||
+ MII_BCM54XX_RDB_ADDR, rdb);
|
||||
if (ret < 0)
|
||||
goto out;
|
||||
|
||||
- ret = __phy_package_write(phydev, MII_BCM54XX_RDB_DATA, val);
|
||||
+ ret = __phy_package_write(phydev, BCM54140_BASE_ADDR,
|
||||
+ MII_BCM54XX_RDB_DATA, val);
|
||||
|
||||
out:
|
||||
phy_unlock_mdio_bus(phydev);
|
||||
--- a/drivers/net/phy/mscc/mscc.h
|
||||
+++ b/drivers/net/phy/mscc/mscc.h
|
||||
@@ -416,6 +416,11 @@ struct vsc8531_private {
|
||||
* gpio_lock: used for PHC operations. Common for all PHYs as the load/save GPIO
|
||||
* is shared.
|
||||
*/
|
||||
+
|
||||
+enum vsc85xx_global_phy {
|
||||
+ VSC88XX_BASE_ADDR = 0,
|
||||
+};
|
||||
+
|
||||
struct vsc85xx_shared_private {
|
||||
struct mutex gpio_lock;
|
||||
};
|
||||
--- a/drivers/net/phy/mscc/mscc_main.c
|
||||
+++ b/drivers/net/phy/mscc/mscc_main.c
|
||||
@@ -711,7 +711,7 @@ int phy_base_write(struct phy_device *ph
|
||||
dump_stack();
|
||||
}
|
||||
|
||||
- return __phy_package_write(phydev, regnum, val);
|
||||
+ return __phy_package_write(phydev, VSC88XX_BASE_ADDR, regnum, val);
|
||||
}
|
||||
|
||||
/* phydev->bus->mdio_lock should be locked when using this function */
|
||||
@@ -722,7 +722,7 @@ int phy_base_read(struct phy_device *phy
|
||||
dump_stack();
|
||||
}
|
||||
|
||||
- return __phy_package_read(phydev, regnum);
|
||||
+ return __phy_package_read(phydev, VSC88XX_BASE_ADDR, regnum);
|
||||
}
|
||||
|
||||
u32 vsc85xx_csr_read(struct phy_device *phydev,
|
||||
--- a/drivers/net/phy/phy_device.c
|
||||
+++ b/drivers/net/phy/phy_device.c
|
||||
@@ -1650,20 +1650,22 @@ EXPORT_SYMBOL_GPL(phy_driver_is_genphy_1
|
||||
/**
|
||||
* phy_package_join - join a common PHY group
|
||||
* @phydev: target phy_device struct
|
||||
- * @addr: cookie and PHY address for global register access
|
||||
+ * @base_addr: cookie and base PHY address of PHY package for offset
|
||||
+ * calculation of global register access
|
||||
* @priv_size: if non-zero allocate this amount of bytes for private data
|
||||
*
|
||||
* This joins a PHY group and provides a shared storage for all phydevs in
|
||||
* this group. This is intended to be used for packages which contain
|
||||
* more than one PHY, for example a quad PHY transceiver.
|
||||
*
|
||||
- * The addr parameter serves as a cookie which has to have the same value
|
||||
- * for all members of one group and as a PHY address to access generic
|
||||
- * registers of a PHY package. Usually, one of the PHY addresses of the
|
||||
- * different PHYs in the package provides access to these global registers.
|
||||
+ * The base_addr parameter serves as cookie which has to have the same values
|
||||
+ * for all members of one group and as the base PHY address of the PHY package
|
||||
+ * for offset calculation to access generic registers of a PHY package.
|
||||
+ * Usually, one of the PHY addresses of the different PHYs in the package
|
||||
+ * provides access to these global registers.
|
||||
* The address which is given here, will be used in the phy_package_read()
|
||||
- * and phy_package_write() convenience functions. If your PHY doesn't have
|
||||
- * global registers you can just pick any of the PHY addresses.
|
||||
+ * and phy_package_write() convenience functions as base and added to the
|
||||
+ * passed offset in those functions.
|
||||
*
|
||||
* This will set the shared pointer of the phydev to the shared storage.
|
||||
* If this is the first call for a this cookie the shared storage will be
|
||||
@@ -1673,17 +1675,17 @@ EXPORT_SYMBOL_GPL(phy_driver_is_genphy_1
|
||||
* Returns < 1 on error, 0 on success. Esp. calling phy_package_join()
|
||||
* with the same cookie but a different priv_size is an error.
|
||||
*/
|
||||
-int phy_package_join(struct phy_device *phydev, int addr, size_t priv_size)
|
||||
+int phy_package_join(struct phy_device *phydev, int base_addr, size_t priv_size)
|
||||
{
|
||||
struct mii_bus *bus = phydev->mdio.bus;
|
||||
struct phy_package_shared *shared;
|
||||
int ret;
|
||||
|
||||
- if (addr < 0 || addr >= PHY_MAX_ADDR)
|
||||
+ if (base_addr < 0 || base_addr >= PHY_MAX_ADDR)
|
||||
return -EINVAL;
|
||||
|
||||
mutex_lock(&bus->shared_lock);
|
||||
- shared = bus->shared[addr];
|
||||
+ shared = bus->shared[base_addr];
|
||||
if (!shared) {
|
||||
ret = -ENOMEM;
|
||||
shared = kzalloc(sizeof(*shared), GFP_KERNEL);
|
||||
@@ -1695,9 +1697,9 @@ int phy_package_join(struct phy_device *
|
||||
goto err_free;
|
||||
shared->priv_size = priv_size;
|
||||
}
|
||||
- shared->addr = addr;
|
||||
+ shared->base_addr = base_addr;
|
||||
refcount_set(&shared->refcnt, 1);
|
||||
- bus->shared[addr] = shared;
|
||||
+ bus->shared[base_addr] = shared;
|
||||
} else {
|
||||
ret = -EINVAL;
|
||||
if (priv_size && priv_size != shared->priv_size)
|
||||
@@ -1735,7 +1737,7 @@ void phy_package_leave(struct phy_device
|
||||
return;
|
||||
|
||||
if (refcount_dec_and_mutex_lock(&shared->refcnt, &bus->shared_lock)) {
|
||||
- bus->shared[shared->addr] = NULL;
|
||||
+ bus->shared[shared->base_addr] = NULL;
|
||||
mutex_unlock(&bus->shared_lock);
|
||||
kfree(shared->priv);
|
||||
kfree(shared);
|
||||
@@ -1754,7 +1756,8 @@ static void devm_phy_package_leave(struc
|
||||
* devm_phy_package_join - resource managed phy_package_join()
|
||||
* @dev: device that is registering this PHY package
|
||||
* @phydev: target phy_device struct
|
||||
- * @addr: cookie and PHY address for global register access
|
||||
+ * @base_addr: cookie and base PHY address of PHY package for offset
|
||||
+ * calculation of global register access
|
||||
* @priv_size: if non-zero allocate this amount of bytes for private data
|
||||
*
|
||||
* Managed phy_package_join(). Shared storage fetched by this function,
|
||||
@@ -1762,7 +1765,7 @@ static void devm_phy_package_leave(struc
|
||||
* phy_package_join() for more information.
|
||||
*/
|
||||
int devm_phy_package_join(struct device *dev, struct phy_device *phydev,
|
||||
- int addr, size_t priv_size)
|
||||
+ int base_addr, size_t priv_size)
|
||||
{
|
||||
struct phy_device **ptr;
|
||||
int ret;
|
||||
@@ -1772,7 +1775,7 @@ int devm_phy_package_join(struct device
|
||||
if (!ptr)
|
||||
return -ENOMEM;
|
||||
|
||||
- ret = phy_package_join(phydev, addr, priv_size);
|
||||
+ ret = phy_package_join(phydev, base_addr, priv_size);
|
||||
|
||||
if (!ret) {
|
||||
*ptr = phydev;
|
||||
--- a/include/linux/phy.h
|
||||
+++ b/include/linux/phy.h
|
||||
@@ -327,7 +327,8 @@ struct mdio_bus_stats {
|
||||
|
||||
/**
|
||||
* struct phy_package_shared - Shared information in PHY packages
|
||||
- * @addr: Common PHY address used to combine PHYs in one package
|
||||
+ * @base_addr: Base PHY address of PHY package used to combine PHYs
|
||||
+ * in one package and for offset calculation of phy_package_read/write
|
||||
* @refcnt: Number of PHYs connected to this shared data
|
||||
* @flags: Initialization of PHY package
|
||||
* @priv_size: Size of the shared private data @priv
|
||||
@@ -338,7 +339,7 @@ struct mdio_bus_stats {
|
||||
* phy_package_leave().
|
||||
*/
|
||||
struct phy_package_shared {
|
||||
- u8 addr;
|
||||
+ u8 base_addr;
|
||||
refcount_t refcnt;
|
||||
unsigned long flags;
|
||||
size_t priv_size;
|
||||
@@ -1969,10 +1970,10 @@ int phy_ethtool_get_link_ksettings(struc
|
||||
int phy_ethtool_set_link_ksettings(struct net_device *ndev,
|
||||
const struct ethtool_link_ksettings *cmd);
|
||||
int phy_ethtool_nway_reset(struct net_device *ndev);
|
||||
-int phy_package_join(struct phy_device *phydev, int addr, size_t priv_size);
|
||||
+int phy_package_join(struct phy_device *phydev, int base_addr, size_t priv_size);
|
||||
void phy_package_leave(struct phy_device *phydev);
|
||||
int devm_phy_package_join(struct device *dev, struct phy_device *phydev,
|
||||
- int addr, size_t priv_size);
|
||||
+ int base_addr, size_t priv_size);
|
||||
|
||||
int __init mdio_bus_init(void);
|
||||
void mdio_bus_exit(void);
|
||||
@@ -1995,46 +1996,65 @@ int __phy_hwtstamp_set(struct phy_device
|
||||
struct kernel_hwtstamp_config *config,
|
||||
struct netlink_ext_ack *extack);
|
||||
|
||||
-static inline int phy_package_read(struct phy_device *phydev, u32 regnum)
|
||||
+static inline int phy_package_address(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset)
|
||||
{
|
||||
struct phy_package_shared *shared = phydev->shared;
|
||||
+ u8 base_addr = shared->base_addr;
|
||||
|
||||
- if (!shared)
|
||||
+ if (addr_offset >= PHY_MAX_ADDR - base_addr)
|
||||
return -EIO;
|
||||
|
||||
- return mdiobus_read(phydev->mdio.bus, shared->addr, regnum);
|
||||
+ /* we know that addr will be in the range 0..31 and thus the
|
||||
+ * implicit cast to a signed int is not a problem.
|
||||
+ */
|
||||
+ return base_addr + addr_offset;
|
||||
}
|
||||
|
||||
-static inline int __phy_package_read(struct phy_device *phydev, u32 regnum)
|
||||
+static inline int phy_package_read(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, u32 regnum)
|
||||
{
|
||||
- struct phy_package_shared *shared = phydev->shared;
|
||||
+ int addr = phy_package_address(phydev, addr_offset);
|
||||
|
||||
- if (!shared)
|
||||
- return -EIO;
|
||||
+ if (addr < 0)
|
||||
+ return addr;
|
||||
+
|
||||
+ return mdiobus_read(phydev->mdio.bus, addr, regnum);
|
||||
+}
|
||||
+
|
||||
+static inline int __phy_package_read(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, u32 regnum)
|
||||
+{
|
||||
+ int addr = phy_package_address(phydev, addr_offset);
|
||||
+
|
||||
+ if (addr < 0)
|
||||
+ return addr;
|
||||
|
||||
- return __mdiobus_read(phydev->mdio.bus, shared->addr, regnum);
|
||||
+ return __mdiobus_read(phydev->mdio.bus, addr, regnum);
|
||||
}
|
||||
|
||||
static inline int phy_package_write(struct phy_device *phydev,
|
||||
- u32 regnum, u16 val)
|
||||
+ unsigned int addr_offset, u32 regnum,
|
||||
+ u16 val)
|
||||
{
|
||||
- struct phy_package_shared *shared = phydev->shared;
|
||||
+ int addr = phy_package_address(phydev, addr_offset);
|
||||
|
||||
- if (!shared)
|
||||
- return -EIO;
|
||||
+ if (addr < 0)
|
||||
+ return addr;
|
||||
|
||||
- return mdiobus_write(phydev->mdio.bus, shared->addr, regnum, val);
|
||||
+ return mdiobus_write(phydev->mdio.bus, addr, regnum, val);
|
||||
}
|
||||
|
||||
static inline int __phy_package_write(struct phy_device *phydev,
|
||||
- u32 regnum, u16 val)
|
||||
+ unsigned int addr_offset, u32 regnum,
|
||||
+ u16 val)
|
||||
{
|
||||
- struct phy_package_shared *shared = phydev->shared;
|
||||
+ int addr = phy_package_address(phydev, addr_offset);
|
||||
|
||||
- if (!shared)
|
||||
- return -EIO;
|
||||
+ if (addr < 0)
|
||||
+ return addr;
|
||||
|
||||
- return __mdiobus_write(phydev->mdio.bus, shared->addr, regnum, val);
|
||||
+ return __mdiobus_write(phydev->mdio.bus, addr, regnum, val);
|
||||
}
|
||||
|
||||
static inline bool __phy_package_set_once(struct phy_device *phydev,
|
@ -0,0 +1,116 @@
|
||||
From 028672bd1d73cf65249a420c1de75e8d2acd2f6a Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 15 Dec 2023 14:15:33 +0100
|
||||
Subject: [PATCH 3/4] net: phy: restructure __phy_write/read_mmd to helper and
|
||||
phydev user
|
||||
|
||||
Restructure phy_write_mmd and phy_read_mmd to implement generic helper
|
||||
for direct mdiobus access for mmd and use these helper for phydev user.
|
||||
|
||||
This is needed in preparation of PHY package API that requires generic
|
||||
access to the mdiobus and are deatched from phydev struct but instead
|
||||
access them based on PHY package base_addr and offsets.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Russell King (Oracle) <rmk+kernel@armlinux.org.uk>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/phy-core.c | 64 ++++++++++++++++++--------------------
|
||||
1 file changed, 30 insertions(+), 34 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/phy-core.c
|
||||
+++ b/drivers/net/phy/phy-core.c
|
||||
@@ -540,6 +540,28 @@ static void mmd_phy_indirect(struct mii_
|
||||
devad | MII_MMD_CTRL_NOINCR);
|
||||
}
|
||||
|
||||
+static int mmd_phy_read(struct mii_bus *bus, int phy_addr, bool is_c45,
|
||||
+ int devad, u32 regnum)
|
||||
+{
|
||||
+ if (is_c45)
|
||||
+ return __mdiobus_c45_read(bus, phy_addr, devad, regnum);
|
||||
+
|
||||
+ mmd_phy_indirect(bus, phy_addr, devad, regnum);
|
||||
+ /* Read the content of the MMD's selected register */
|
||||
+ return __mdiobus_read(bus, phy_addr, MII_MMD_DATA);
|
||||
+}
|
||||
+
|
||||
+static int mmd_phy_write(struct mii_bus *bus, int phy_addr, bool is_c45,
|
||||
+ int devad, u32 regnum, u16 val)
|
||||
+{
|
||||
+ if (is_c45)
|
||||
+ return __mdiobus_c45_write(bus, phy_addr, devad, regnum, val);
|
||||
+
|
||||
+ mmd_phy_indirect(bus, phy_addr, devad, regnum);
|
||||
+ /* Write the data into MMD's selected register */
|
||||
+ return __mdiobus_write(bus, phy_addr, MII_MMD_DATA, val);
|
||||
+}
|
||||
+
|
||||
/**
|
||||
* __phy_read_mmd - Convenience function for reading a register
|
||||
* from an MMD on a given PHY.
|
||||
@@ -551,26 +573,14 @@ static void mmd_phy_indirect(struct mii_
|
||||
*/
|
||||
int __phy_read_mmd(struct phy_device *phydev, int devad, u32 regnum)
|
||||
{
|
||||
- int val;
|
||||
-
|
||||
if (regnum > (u16)~0 || devad > 32)
|
||||
return -EINVAL;
|
||||
|
||||
- if (phydev->drv && phydev->drv->read_mmd) {
|
||||
- val = phydev->drv->read_mmd(phydev, devad, regnum);
|
||||
- } else if (phydev->is_c45) {
|
||||
- val = __mdiobus_c45_read(phydev->mdio.bus, phydev->mdio.addr,
|
||||
- devad, regnum);
|
||||
- } else {
|
||||
- struct mii_bus *bus = phydev->mdio.bus;
|
||||
- int phy_addr = phydev->mdio.addr;
|
||||
-
|
||||
- mmd_phy_indirect(bus, phy_addr, devad, regnum);
|
||||
-
|
||||
- /* Read the content of the MMD's selected register */
|
||||
- val = __mdiobus_read(bus, phy_addr, MII_MMD_DATA);
|
||||
- }
|
||||
- return val;
|
||||
+ if (phydev->drv && phydev->drv->read_mmd)
|
||||
+ return phydev->drv->read_mmd(phydev, devad, regnum);
|
||||
+
|
||||
+ return mmd_phy_read(phydev->mdio.bus, phydev->mdio.addr,
|
||||
+ phydev->is_c45, devad, regnum);
|
||||
}
|
||||
EXPORT_SYMBOL(__phy_read_mmd);
|
||||
|
||||
@@ -607,28 +617,14 @@ EXPORT_SYMBOL(phy_read_mmd);
|
||||
*/
|
||||
int __phy_write_mmd(struct phy_device *phydev, int devad, u32 regnum, u16 val)
|
||||
{
|
||||
- int ret;
|
||||
-
|
||||
if (regnum > (u16)~0 || devad > 32)
|
||||
return -EINVAL;
|
||||
|
||||
- if (phydev->drv && phydev->drv->write_mmd) {
|
||||
- ret = phydev->drv->write_mmd(phydev, devad, regnum, val);
|
||||
- } else if (phydev->is_c45) {
|
||||
- ret = __mdiobus_c45_write(phydev->mdio.bus, phydev->mdio.addr,
|
||||
- devad, regnum, val);
|
||||
- } else {
|
||||
- struct mii_bus *bus = phydev->mdio.bus;
|
||||
- int phy_addr = phydev->mdio.addr;
|
||||
+ if (phydev->drv && phydev->drv->write_mmd)
|
||||
+ return phydev->drv->write_mmd(phydev, devad, regnum, val);
|
||||
|
||||
- mmd_phy_indirect(bus, phy_addr, devad, regnum);
|
||||
-
|
||||
- /* Write the data into MMD's selected register */
|
||||
- __mdiobus_write(bus, phy_addr, MII_MMD_DATA, val);
|
||||
-
|
||||
- ret = 0;
|
||||
- }
|
||||
- return ret;
|
||||
+ return mmd_phy_write(phydev->mdio.bus, phydev->mdio.addr,
|
||||
+ phydev->is_c45, devad, regnum, val);
|
||||
}
|
||||
EXPORT_SYMBOL(__phy_write_mmd);
|
||||
|
@ -0,0 +1,196 @@
|
||||
From d63710fc0f1a501fd75a7025e3070a96ffa1645f Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Fri, 15 Dec 2023 14:15:34 +0100
|
||||
Subject: [PATCH 4/4] net: phy: add support for PHY package MMD read/write
|
||||
|
||||
Some PHY in PHY package may require to read/write MMD regs to correctly
|
||||
configure the PHY package.
|
||||
|
||||
Add support for these additional required function in both lock and no
|
||||
lock variant.
|
||||
|
||||
It's assumed that the entire PHY package is either C22 or C45. We use
|
||||
C22 or C45 way of writing/reading to mmd regs based on the passed phydev
|
||||
whether it's C22 or C45.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/phy-core.c | 140 +++++++++++++++++++++++++++++++++++++
|
||||
include/linux/phy.h | 16 +++++
|
||||
2 files changed, 156 insertions(+)
|
||||
|
||||
--- a/drivers/net/phy/phy-core.c
|
||||
+++ b/drivers/net/phy/phy-core.c
|
||||
@@ -651,6 +651,146 @@ int phy_write_mmd(struct phy_device *phy
|
||||
EXPORT_SYMBOL(phy_write_mmd);
|
||||
|
||||
/**
|
||||
+ * __phy_package_read_mmd - read MMD reg relative to PHY package base addr
|
||||
+ * @phydev: The phy_device struct
|
||||
+ * @addr_offset: The offset to be added to PHY package base_addr
|
||||
+ * @devad: The MMD to read from
|
||||
+ * @regnum: The register on the MMD to read
|
||||
+ *
|
||||
+ * Convenience helper for reading a register of an MMD on a given PHY
|
||||
+ * using the PHY package base address. The base address is added to
|
||||
+ * the addr_offset value.
|
||||
+ *
|
||||
+ * Same calling rules as for __phy_read();
|
||||
+ *
|
||||
+ * NOTE: It's assumed that the entire PHY package is either C22 or C45.
|
||||
+ */
|
||||
+int __phy_package_read_mmd(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, int devad,
|
||||
+ u32 regnum)
|
||||
+{
|
||||
+ int addr = phy_package_address(phydev, addr_offset);
|
||||
+
|
||||
+ if (addr < 0)
|
||||
+ return addr;
|
||||
+
|
||||
+ if (regnum > (u16)~0 || devad > 32)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ return mmd_phy_read(phydev->mdio.bus, addr, phydev->is_c45, devad,
|
||||
+ regnum);
|
||||
+}
|
||||
+EXPORT_SYMBOL(__phy_package_read_mmd);
|
||||
+
|
||||
+/**
|
||||
+ * phy_package_read_mmd - read MMD reg relative to PHY package base addr
|
||||
+ * @phydev: The phy_device struct
|
||||
+ * @addr_offset: The offset to be added to PHY package base_addr
|
||||
+ * @devad: The MMD to read from
|
||||
+ * @regnum: The register on the MMD to read
|
||||
+ *
|
||||
+ * Convenience helper for reading a register of an MMD on a given PHY
|
||||
+ * using the PHY package base address. The base address is added to
|
||||
+ * the addr_offset value.
|
||||
+ *
|
||||
+ * Same calling rules as for phy_read();
|
||||
+ *
|
||||
+ * NOTE: It's assumed that the entire PHY package is either C22 or C45.
|
||||
+ */
|
||||
+int phy_package_read_mmd(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, int devad,
|
||||
+ u32 regnum)
|
||||
+{
|
||||
+ int addr = phy_package_address(phydev, addr_offset);
|
||||
+ int val;
|
||||
+
|
||||
+ if (addr < 0)
|
||||
+ return addr;
|
||||
+
|
||||
+ if (regnum > (u16)~0 || devad > 32)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ phy_lock_mdio_bus(phydev);
|
||||
+ val = mmd_phy_read(phydev->mdio.bus, addr, phydev->is_c45, devad,
|
||||
+ regnum);
|
||||
+ phy_unlock_mdio_bus(phydev);
|
||||
+
|
||||
+ return val;
|
||||
+}
|
||||
+EXPORT_SYMBOL(phy_package_read_mmd);
|
||||
+
|
||||
+/**
|
||||
+ * __phy_package_write_mmd - write MMD reg relative to PHY package base addr
|
||||
+ * @phydev: The phy_device struct
|
||||
+ * @addr_offset: The offset to be added to PHY package base_addr
|
||||
+ * @devad: The MMD to write to
|
||||
+ * @regnum: The register on the MMD to write
|
||||
+ * @val: value to write to @regnum
|
||||
+ *
|
||||
+ * Convenience helper for writing a register of an MMD on a given PHY
|
||||
+ * using the PHY package base address. The base address is added to
|
||||
+ * the addr_offset value.
|
||||
+ *
|
||||
+ * Same calling rules as for __phy_write();
|
||||
+ *
|
||||
+ * NOTE: It's assumed that the entire PHY package is either C22 or C45.
|
||||
+ */
|
||||
+int __phy_package_write_mmd(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, int devad,
|
||||
+ u32 regnum, u16 val)
|
||||
+{
|
||||
+ int addr = phy_package_address(phydev, addr_offset);
|
||||
+
|
||||
+ if (addr < 0)
|
||||
+ return addr;
|
||||
+
|
||||
+ if (regnum > (u16)~0 || devad > 32)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ return mmd_phy_write(phydev->mdio.bus, addr, phydev->is_c45, devad,
|
||||
+ regnum, val);
|
||||
+}
|
||||
+EXPORT_SYMBOL(__phy_package_write_mmd);
|
||||
+
|
||||
+/**
|
||||
+ * phy_package_write_mmd - write MMD reg relative to PHY package base addr
|
||||
+ * @phydev: The phy_device struct
|
||||
+ * @addr_offset: The offset to be added to PHY package base_addr
|
||||
+ * @devad: The MMD to write to
|
||||
+ * @regnum: The register on the MMD to write
|
||||
+ * @val: value to write to @regnum
|
||||
+ *
|
||||
+ * Convenience helper for writing a register of an MMD on a given PHY
|
||||
+ * using the PHY package base address. The base address is added to
|
||||
+ * the addr_offset value.
|
||||
+ *
|
||||
+ * Same calling rules as for phy_write();
|
||||
+ *
|
||||
+ * NOTE: It's assumed that the entire PHY package is either C22 or C45.
|
||||
+ */
|
||||
+int phy_package_write_mmd(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, int devad,
|
||||
+ u32 regnum, u16 val)
|
||||
+{
|
||||
+ int addr = phy_package_address(phydev, addr_offset);
|
||||
+ int ret;
|
||||
+
|
||||
+ if (addr < 0)
|
||||
+ return addr;
|
||||
+
|
||||
+ if (regnum > (u16)~0 || devad > 32)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ phy_lock_mdio_bus(phydev);
|
||||
+ ret = mmd_phy_write(phydev->mdio.bus, addr, phydev->is_c45, devad,
|
||||
+ regnum, val);
|
||||
+ phy_unlock_mdio_bus(phydev);
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+EXPORT_SYMBOL(phy_package_write_mmd);
|
||||
+
|
||||
+/**
|
||||
* phy_modify_changed - Function for modifying a PHY register
|
||||
* @phydev: the phy_device struct
|
||||
* @regnum: register number to modify
|
||||
--- a/include/linux/phy.h
|
||||
+++ b/include/linux/phy.h
|
||||
@@ -2057,6 +2057,22 @@ static inline int __phy_package_write(st
|
||||
return __mdiobus_write(phydev->mdio.bus, addr, regnum, val);
|
||||
}
|
||||
|
||||
+int __phy_package_read_mmd(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, int devad,
|
||||
+ u32 regnum);
|
||||
+
|
||||
+int phy_package_read_mmd(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, int devad,
|
||||
+ u32 regnum);
|
||||
+
|
||||
+int __phy_package_write_mmd(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, int devad,
|
||||
+ u32 regnum, u16 val);
|
||||
+
|
||||
+int phy_package_write_mmd(struct phy_device *phydev,
|
||||
+ unsigned int addr_offset, int devad,
|
||||
+ u32 regnum, u16 val);
|
||||
+
|
||||
static inline bool __phy_package_set_once(struct phy_device *phydev,
|
||||
unsigned int b)
|
||||
{
|
@ -0,0 +1,36 @@
|
||||
From f2ec98566775dd4341ec1dcf93aa5859c60de826 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 1 Feb 2024 14:46:00 +0100
|
||||
Subject: [PATCH 1/2] net: phy: qcom: qca808x: fix logic error in LED
|
||||
brightness set
|
||||
|
||||
In switching to using phy_modify_mmd and a more short version of the
|
||||
LED ON/OFF condition in later revision, it was made a logic error where
|
||||
|
||||
value ? QCA808X_LED_FORCE_ON : QCA808X_LED_FORCE_OFF is always true as
|
||||
value is always OR with QCA808X_LED_FORCE_EN due to missing ()
|
||||
resulting in the testing condition being QCA808X_LED_FORCE_EN | value.
|
||||
|
||||
Add the () to apply the correct condition and restore correct
|
||||
functionality of the brightness ON/OFF.
|
||||
|
||||
Fixes: 7196062b64ee ("net: phy: at803x: add LED support for qca808x")
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/qcom/qca808x.c | 4 ++--
|
||||
1 file changed, 2 insertions(+), 2 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/qcom/qca808x.c
|
||||
+++ b/drivers/net/phy/qcom/qca808x.c
|
||||
@@ -820,8 +820,8 @@ static int qca808x_led_brightness_set(st
|
||||
|
||||
return phy_modify_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_MODE_MASK,
|
||||
- QCA808X_LED_FORCE_EN | value ? QCA808X_LED_FORCE_ON :
|
||||
- QCA808X_LED_FORCE_OFF);
|
||||
+ QCA808X_LED_FORCE_EN | (value ? QCA808X_LED_FORCE_ON :
|
||||
+ QCA808X_LED_FORCE_OFF));
|
||||
}
|
||||
|
||||
static int qca808x_led_blink_set(struct phy_device *phydev, u8 index,
|
@ -0,0 +1,41 @@
|
||||
From f203c8c77c7616c099647636f4c67d59a45fe8a2 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 1 Feb 2024 14:46:01 +0100
|
||||
Subject: [PATCH 2/2] net: phy: qcom: qca808x: default to LED active High if
|
||||
not set
|
||||
|
||||
qca808x PHY provide support for the led_polarity_set OP to configure
|
||||
and apply the active-low property but on PHY reset, the Active High bit
|
||||
is not set resulting in the LED driven as active-low.
|
||||
|
||||
To fix this, check if active-low is not set in DT and enable Active High
|
||||
polarity by default to restore correct funcionality of the LED.
|
||||
|
||||
Fixes: 7196062b64ee ("net: phy: at803x: add LED support for qca808x")
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/qcom/qca808x.c | 10 ++++++++++
|
||||
1 file changed, 10 insertions(+)
|
||||
|
||||
--- a/drivers/net/phy/qcom/qca808x.c
|
||||
+++ b/drivers/net/phy/qcom/qca808x.c
|
||||
@@ -290,8 +290,18 @@ static int qca808x_probe(struct phy_devi
|
||||
|
||||
static int qca808x_config_init(struct phy_device *phydev)
|
||||
{
|
||||
+ struct qca808x_priv *priv = phydev->priv;
|
||||
int ret;
|
||||
|
||||
+ /* Default to LED Active High if active-low not in DT */
|
||||
+ if (priv->led_polarity_mode == -1) {
|
||||
+ ret = phy_set_bits_mmd(phydev, MDIO_MMD_AN,
|
||||
+ QCA808X_MMD7_LED_POLARITY_CTRL,
|
||||
+ QCA808X_LED_ACTIVE_HIGH);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
/* Active adc&vga on 802.3az for the link 1000M and 100M */
|
||||
ret = phy_modify_mmd(phydev, MDIO_MMD_PCS, QCA808X_PHY_MMD3_ADDR_CLD_CTRL7,
|
||||
QCA808X_8023AZ_AFE_CTRL_MASK, QCA808X_8023AZ_AFE_EN);
|
@ -0,0 +1,211 @@
|
||||
From 385ef48f468696d6d172eb367656a3466fa0408d Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 6 Feb 2024 18:31:05 +0100
|
||||
Subject: [PATCH 02/10] net: phy: add support for scanning PHY in PHY packages
|
||||
nodes
|
||||
|
||||
Add support for scanning PHY in PHY package nodes. PHY packages nodes
|
||||
are just container for actual PHY on the MDIO bus.
|
||||
|
||||
Their PHY address defined in the PHY package node are absolute and
|
||||
reflect the address on the MDIO bus.
|
||||
|
||||
mdio_bus.c and of_mdio.c is updated to now support and parse also
|
||||
PHY package subnode by checking if the node name match
|
||||
"ethernet-phy-package".
|
||||
|
||||
As PHY package reg is mandatory and each PHY in the PHY package must
|
||||
have a reg, every invalid PHY Package node is ignored and will be
|
||||
skipped by the autoscan fallback.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/mdio/of_mdio.c | 79 +++++++++++++++++++++++++++-----------
|
||||
drivers/net/phy/mdio_bus.c | 44 +++++++++++++++++----
|
||||
2 files changed, 92 insertions(+), 31 deletions(-)
|
||||
|
||||
--- a/drivers/net/mdio/of_mdio.c
|
||||
+++ b/drivers/net/mdio/of_mdio.c
|
||||
@@ -138,6 +138,53 @@ bool of_mdiobus_child_is_phy(struct devi
|
||||
}
|
||||
EXPORT_SYMBOL(of_mdiobus_child_is_phy);
|
||||
|
||||
+static int __of_mdiobus_parse_phys(struct mii_bus *mdio, struct device_node *np,
|
||||
+ bool *scanphys)
|
||||
+{
|
||||
+ struct device_node *child;
|
||||
+ int addr, rc = 0;
|
||||
+
|
||||
+ /* Loop over the child nodes and register a phy_device for each phy */
|
||||
+ for_each_available_child_of_node(np, child) {
|
||||
+ if (of_node_name_eq(child, "ethernet-phy-package")) {
|
||||
+ /* Ignore invalid ethernet-phy-package node */
|
||||
+ if (!of_find_property(child, "reg", NULL))
|
||||
+ continue;
|
||||
+
|
||||
+ rc = __of_mdiobus_parse_phys(mdio, child, NULL);
|
||||
+ if (rc && rc != -ENODEV)
|
||||
+ goto exit;
|
||||
+
|
||||
+ continue;
|
||||
+ }
|
||||
+
|
||||
+ addr = of_mdio_parse_addr(&mdio->dev, child);
|
||||
+ if (addr < 0) {
|
||||
+ /* Skip scanning for invalid ethernet-phy-package node */
|
||||
+ if (scanphys)
|
||||
+ *scanphys = true;
|
||||
+ continue;
|
||||
+ }
|
||||
+
|
||||
+ if (of_mdiobus_child_is_phy(child))
|
||||
+ rc = of_mdiobus_register_phy(mdio, child, addr);
|
||||
+ else
|
||||
+ rc = of_mdiobus_register_device(mdio, child, addr);
|
||||
+
|
||||
+ if (rc == -ENODEV)
|
||||
+ dev_err(&mdio->dev,
|
||||
+ "MDIO device at address %d is missing.\n",
|
||||
+ addr);
|
||||
+ else if (rc)
|
||||
+ goto exit;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+exit:
|
||||
+ of_node_put(child);
|
||||
+ return rc;
|
||||
+}
|
||||
+
|
||||
/**
|
||||
* __of_mdiobus_register - Register mii_bus and create PHYs from the device tree
|
||||
* @mdio: pointer to mii_bus structure
|
||||
@@ -179,33 +226,18 @@ int __of_mdiobus_register(struct mii_bus
|
||||
return rc;
|
||||
|
||||
/* Loop over the child nodes and register a phy_device for each phy */
|
||||
- for_each_available_child_of_node(np, child) {
|
||||
- addr = of_mdio_parse_addr(&mdio->dev, child);
|
||||
- if (addr < 0) {
|
||||
- scanphys = true;
|
||||
- continue;
|
||||
- }
|
||||
-
|
||||
- if (of_mdiobus_child_is_phy(child))
|
||||
- rc = of_mdiobus_register_phy(mdio, child, addr);
|
||||
- else
|
||||
- rc = of_mdiobus_register_device(mdio, child, addr);
|
||||
-
|
||||
- if (rc == -ENODEV)
|
||||
- dev_err(&mdio->dev,
|
||||
- "MDIO device at address %d is missing.\n",
|
||||
- addr);
|
||||
- else if (rc)
|
||||
- goto unregister;
|
||||
- }
|
||||
+ rc = __of_mdiobus_parse_phys(mdio, np, &scanphys);
|
||||
+ if (rc)
|
||||
+ goto unregister;
|
||||
|
||||
if (!scanphys)
|
||||
return 0;
|
||||
|
||||
/* auto scan for PHYs with empty reg property */
|
||||
for_each_available_child_of_node(np, child) {
|
||||
- /* Skip PHYs with reg property set */
|
||||
- if (of_property_present(child, "reg"))
|
||||
+ /* Skip PHYs with reg property set or ethernet-phy-package node */
|
||||
+ if (of_property_present(child, "reg") ||
|
||||
+ of_node_name_eq(child, "ethernet-phy-package"))
|
||||
continue;
|
||||
|
||||
for (addr = 0; addr < PHY_MAX_ADDR; addr++) {
|
||||
@@ -226,15 +258,16 @@ int __of_mdiobus_register(struct mii_bus
|
||||
if (!rc)
|
||||
break;
|
||||
if (rc != -ENODEV)
|
||||
- goto unregister;
|
||||
+ goto put_unregister;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
||||
-unregister:
|
||||
+put_unregister:
|
||||
of_node_put(child);
|
||||
+unregister:
|
||||
mdiobus_unregister(mdio);
|
||||
return rc;
|
||||
}
|
||||
--- a/drivers/net/phy/mdio_bus.c
|
||||
+++ b/drivers/net/phy/mdio_bus.c
|
||||
@@ -455,19 +455,34 @@ EXPORT_SYMBOL(of_mdio_find_bus);
|
||||
* found, set the of_node pointer for the mdio device. This allows
|
||||
* auto-probed phy devices to be supplied with information passed in
|
||||
* via DT.
|
||||
+ * If a PHY package is found, PHY is searched also there.
|
||||
*/
|
||||
-static void of_mdiobus_link_mdiodev(struct mii_bus *bus,
|
||||
- struct mdio_device *mdiodev)
|
||||
+static int of_mdiobus_find_phy(struct device *dev, struct mdio_device *mdiodev,
|
||||
+ struct device_node *np)
|
||||
{
|
||||
- struct device *dev = &mdiodev->dev;
|
||||
struct device_node *child;
|
||||
|
||||
- if (dev->of_node || !bus->dev.of_node)
|
||||
- return;
|
||||
-
|
||||
- for_each_available_child_of_node(bus->dev.of_node, child) {
|
||||
+ for_each_available_child_of_node(np, child) {
|
||||
int addr;
|
||||
|
||||
+ if (of_node_name_eq(child, "ethernet-phy-package")) {
|
||||
+ /* Validate PHY package reg presence */
|
||||
+ if (!of_find_property(child, "reg", NULL)) {
|
||||
+ of_node_put(child);
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ if (!of_mdiobus_find_phy(dev, mdiodev, child)) {
|
||||
+ /* The refcount for the PHY package will be
|
||||
+ * incremented later when PHY join the Package.
|
||||
+ */
|
||||
+ of_node_put(child);
|
||||
+ return 0;
|
||||
+ }
|
||||
+
|
||||
+ continue;
|
||||
+ }
|
||||
+
|
||||
addr = of_mdio_parse_addr(dev, child);
|
||||
if (addr < 0)
|
||||
continue;
|
||||
@@ -477,9 +492,22 @@ static void of_mdiobus_link_mdiodev(stru
|
||||
/* The refcount on "child" is passed to the mdio
|
||||
* device. Do _not_ use of_node_put(child) here.
|
||||
*/
|
||||
- return;
|
||||
+ return 0;
|
||||
}
|
||||
}
|
||||
+
|
||||
+ return -ENODEV;
|
||||
+}
|
||||
+
|
||||
+static void of_mdiobus_link_mdiodev(struct mii_bus *bus,
|
||||
+ struct mdio_device *mdiodev)
|
||||
+{
|
||||
+ struct device *dev = &mdiodev->dev;
|
||||
+
|
||||
+ if (dev->of_node || !bus->dev.of_node)
|
||||
+ return;
|
||||
+
|
||||
+ of_mdiobus_find_phy(dev, mdiodev, bus->dev.of_node);
|
||||
}
|
||||
#else /* !IS_ENABLED(CONFIG_OF_MDIO) */
|
||||
static inline void of_mdiobus_link_mdiodev(struct mii_bus *mdio,
|
@ -0,0 +1,185 @@
|
||||
From 471e8fd3afcef5a9f9089f0bd21965ad9ba35c91 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 6 Feb 2024 18:31:06 +0100
|
||||
Subject: [PATCH 03/10] net: phy: add devm/of_phy_package_join helper
|
||||
|
||||
Add devm/of_phy_package_join helper to join PHYs in a PHY package. These
|
||||
are variant of the manual phy_package_join with the difference that
|
||||
these will use DT nodes to derive the base_addr instead of manually
|
||||
passing an hardcoded value.
|
||||
|
||||
An additional value is added in phy_package_shared, "np" to reference
|
||||
the PHY package node pointer in specific PHY driver probe_once and
|
||||
config_init_once functions to make use of additional specific properties
|
||||
defined in the PHY package node in DT.
|
||||
|
||||
The np value is filled only with of_phy_package_join if a valid PHY
|
||||
package node is found. A valid PHY package node must have the node name
|
||||
set to "ethernet-phy-package".
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/phy_device.c | 96 ++++++++++++++++++++++++++++++++++++
|
||||
include/linux/phy.h | 6 +++
|
||||
2 files changed, 102 insertions(+)
|
||||
|
||||
--- a/drivers/net/phy/phy_device.c
|
||||
+++ b/drivers/net/phy/phy_device.c
|
||||
@@ -1698,6 +1698,7 @@ int phy_package_join(struct phy_device *
|
||||
shared->priv_size = priv_size;
|
||||
}
|
||||
shared->base_addr = base_addr;
|
||||
+ shared->np = NULL;
|
||||
refcount_set(&shared->refcnt, 1);
|
||||
bus->shared[base_addr] = shared;
|
||||
} else {
|
||||
@@ -1721,6 +1722,63 @@ err_unlock:
|
||||
EXPORT_SYMBOL_GPL(phy_package_join);
|
||||
|
||||
/**
|
||||
+ * of_phy_package_join - join a common PHY group in PHY package
|
||||
+ * @phydev: target phy_device struct
|
||||
+ * @priv_size: if non-zero allocate this amount of bytes for private data
|
||||
+ *
|
||||
+ * This is a variant of phy_package_join for PHY package defined in DT.
|
||||
+ *
|
||||
+ * The parent node of the @phydev is checked as a valid PHY package node
|
||||
+ * structure (by matching the node name "ethernet-phy-package") and the
|
||||
+ * base_addr for the PHY package is passed to phy_package_join.
|
||||
+ *
|
||||
+ * With this configuration the shared struct will also have the np value
|
||||
+ * filled to use additional DT defined properties in PHY specific
|
||||
+ * probe_once and config_init_once PHY package OPs.
|
||||
+ *
|
||||
+ * Returns < 0 on error, 0 on success. Esp. calling phy_package_join()
|
||||
+ * with the same cookie but a different priv_size is an error. Or a parent
|
||||
+ * node is not detected or is not valid or doesn't match the expected node
|
||||
+ * name for PHY package.
|
||||
+ */
|
||||
+int of_phy_package_join(struct phy_device *phydev, size_t priv_size)
|
||||
+{
|
||||
+ struct device_node *node = phydev->mdio.dev.of_node;
|
||||
+ struct device_node *package_node;
|
||||
+ u32 base_addr;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (!node)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ package_node = of_get_parent(node);
|
||||
+ if (!package_node)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ if (!of_node_name_eq(package_node, "ethernet-phy-package")) {
|
||||
+ ret = -EINVAL;
|
||||
+ goto exit;
|
||||
+ }
|
||||
+
|
||||
+ if (of_property_read_u32(package_node, "reg", &base_addr)) {
|
||||
+ ret = -EINVAL;
|
||||
+ goto exit;
|
||||
+ }
|
||||
+
|
||||
+ ret = phy_package_join(phydev, base_addr, priv_size);
|
||||
+ if (ret)
|
||||
+ goto exit;
|
||||
+
|
||||
+ phydev->shared->np = package_node;
|
||||
+
|
||||
+ return 0;
|
||||
+exit:
|
||||
+ of_node_put(package_node);
|
||||
+ return ret;
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(of_phy_package_join);
|
||||
+
|
||||
+/**
|
||||
* phy_package_leave - leave a common PHY group
|
||||
* @phydev: target phy_device struct
|
||||
*
|
||||
@@ -1736,6 +1794,10 @@ void phy_package_leave(struct phy_device
|
||||
if (!shared)
|
||||
return;
|
||||
|
||||
+ /* Decrease the node refcount on leave if present */
|
||||
+ if (shared->np)
|
||||
+ of_node_put(shared->np);
|
||||
+
|
||||
if (refcount_dec_and_mutex_lock(&shared->refcnt, &bus->shared_lock)) {
|
||||
bus->shared[shared->base_addr] = NULL;
|
||||
mutex_unlock(&bus->shared_lock);
|
||||
@@ -1789,6 +1851,40 @@ int devm_phy_package_join(struct device
|
||||
EXPORT_SYMBOL_GPL(devm_phy_package_join);
|
||||
|
||||
/**
|
||||
+ * devm_of_phy_package_join - resource managed of_phy_package_join()
|
||||
+ * @dev: device that is registering this PHY package
|
||||
+ * @phydev: target phy_device struct
|
||||
+ * @priv_size: if non-zero allocate this amount of bytes for private data
|
||||
+ *
|
||||
+ * Managed of_phy_package_join(). Shared storage fetched by this function,
|
||||
+ * phy_package_leave() is automatically called on driver detach. See
|
||||
+ * of_phy_package_join() for more information.
|
||||
+ */
|
||||
+int devm_of_phy_package_join(struct device *dev, struct phy_device *phydev,
|
||||
+ size_t priv_size)
|
||||
+{
|
||||
+ struct phy_device **ptr;
|
||||
+ int ret;
|
||||
+
|
||||
+ ptr = devres_alloc(devm_phy_package_leave, sizeof(*ptr),
|
||||
+ GFP_KERNEL);
|
||||
+ if (!ptr)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ ret = of_phy_package_join(phydev, priv_size);
|
||||
+
|
||||
+ if (!ret) {
|
||||
+ *ptr = phydev;
|
||||
+ devres_add(dev, ptr);
|
||||
+ } else {
|
||||
+ devres_free(ptr);
|
||||
+ }
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(devm_of_phy_package_join);
|
||||
+
|
||||
+/**
|
||||
* phy_detach - detach a PHY device from its network device
|
||||
* @phydev: target phy_device struct
|
||||
*
|
||||
--- a/include/linux/phy.h
|
||||
+++ b/include/linux/phy.h
|
||||
@@ -329,6 +329,7 @@ struct mdio_bus_stats {
|
||||
* struct phy_package_shared - Shared information in PHY packages
|
||||
* @base_addr: Base PHY address of PHY package used to combine PHYs
|
||||
* in one package and for offset calculation of phy_package_read/write
|
||||
+ * @np: Pointer to the Device Node if PHY package defined in DT
|
||||
* @refcnt: Number of PHYs connected to this shared data
|
||||
* @flags: Initialization of PHY package
|
||||
* @priv_size: Size of the shared private data @priv
|
||||
@@ -340,6 +341,8 @@ struct mdio_bus_stats {
|
||||
*/
|
||||
struct phy_package_shared {
|
||||
u8 base_addr;
|
||||
+ /* With PHY package defined in DT this points to the PHY package node */
|
||||
+ struct device_node *np;
|
||||
refcount_t refcnt;
|
||||
unsigned long flags;
|
||||
size_t priv_size;
|
||||
@@ -1971,9 +1974,12 @@ int phy_ethtool_set_link_ksettings(struc
|
||||
const struct ethtool_link_ksettings *cmd);
|
||||
int phy_ethtool_nway_reset(struct net_device *ndev);
|
||||
int phy_package_join(struct phy_device *phydev, int base_addr, size_t priv_size);
|
||||
+int of_phy_package_join(struct phy_device *phydev, size_t priv_size);
|
||||
void phy_package_leave(struct phy_device *phydev);
|
||||
int devm_phy_package_join(struct device *dev, struct phy_device *phydev,
|
||||
int base_addr, size_t priv_size);
|
||||
+int devm_of_phy_package_join(struct device *dev, struct phy_device *phydev,
|
||||
+ size_t priv_size);
|
||||
|
||||
int __init mdio_bus_init(void);
|
||||
void mdio_bus_exit(void);
|
@ -0,0 +1,583 @@
|
||||
From 737eb75a815f9c08dcbb6631db57f4f4b0540a5b Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 6 Feb 2024 18:31:07 +0100
|
||||
Subject: [PATCH 04/10] net: phy: qcom: move more function to shared library
|
||||
|
||||
Move more function to shared library in preparation for introduction of
|
||||
new PHY Family qca807x that will make use of both functions from at803x
|
||||
and qca808x as it's a transition PHY with some implementation of at803x
|
||||
and some from the new qca808x.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/qcom/at803x.c | 35 -----
|
||||
drivers/net/phy/qcom/qca808x.c | 205 ----------------------------
|
||||
drivers/net/phy/qcom/qcom-phy-lib.c | 193 ++++++++++++++++++++++++++
|
||||
drivers/net/phy/qcom/qcom.h | 51 +++++++
|
||||
4 files changed, 244 insertions(+), 240 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/qcom/at803x.c
|
||||
+++ b/drivers/net/phy/qcom/at803x.c
|
||||
@@ -504,41 +504,6 @@ static void at803x_link_change_notify(st
|
||||
}
|
||||
}
|
||||
|
||||
-static int at803x_read_status(struct phy_device *phydev)
|
||||
-{
|
||||
- struct at803x_ss_mask ss_mask = { 0 };
|
||||
- int err, old_link = phydev->link;
|
||||
-
|
||||
- /* Update the link, but return if there was an error */
|
||||
- err = genphy_update_link(phydev);
|
||||
- if (err)
|
||||
- return err;
|
||||
-
|
||||
- /* why bother the PHY if nothing can have changed */
|
||||
- if (phydev->autoneg == AUTONEG_ENABLE && old_link && phydev->link)
|
||||
- return 0;
|
||||
-
|
||||
- phydev->speed = SPEED_UNKNOWN;
|
||||
- phydev->duplex = DUPLEX_UNKNOWN;
|
||||
- phydev->pause = 0;
|
||||
- phydev->asym_pause = 0;
|
||||
-
|
||||
- err = genphy_read_lpa(phydev);
|
||||
- if (err < 0)
|
||||
- return err;
|
||||
-
|
||||
- ss_mask.speed_mask = AT803X_SS_SPEED_MASK;
|
||||
- ss_mask.speed_shift = __bf_shf(AT803X_SS_SPEED_MASK);
|
||||
- err = at803x_read_specific_status(phydev, ss_mask);
|
||||
- if (err < 0)
|
||||
- return err;
|
||||
-
|
||||
- if (phydev->autoneg == AUTONEG_ENABLE && phydev->autoneg_complete)
|
||||
- phy_resolve_aneg_pause(phydev);
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
-
|
||||
static int at803x_config_aneg(struct phy_device *phydev)
|
||||
{
|
||||
struct at803x_priv *priv = phydev->priv;
|
||||
--- a/drivers/net/phy/qcom/qca808x.c
|
||||
+++ b/drivers/net/phy/qcom/qca808x.c
|
||||
@@ -2,7 +2,6 @@
|
||||
|
||||
#include <linux/phy.h>
|
||||
#include <linux/module.h>
|
||||
-#include <linux/ethtool_netlink.h>
|
||||
|
||||
#include "qcom.h"
|
||||
|
||||
@@ -63,55 +62,6 @@
|
||||
#define QCA808X_DBG_AN_TEST 0xb
|
||||
#define QCA808X_HIBERNATION_EN BIT(15)
|
||||
|
||||
-#define QCA808X_CDT_ENABLE_TEST BIT(15)
|
||||
-#define QCA808X_CDT_INTER_CHECK_DIS BIT(13)
|
||||
-#define QCA808X_CDT_STATUS BIT(11)
|
||||
-#define QCA808X_CDT_LENGTH_UNIT BIT(10)
|
||||
-
|
||||
-#define QCA808X_MMD3_CDT_STATUS 0x8064
|
||||
-#define QCA808X_MMD3_CDT_DIAG_PAIR_A 0x8065
|
||||
-#define QCA808X_MMD3_CDT_DIAG_PAIR_B 0x8066
|
||||
-#define QCA808X_MMD3_CDT_DIAG_PAIR_C 0x8067
|
||||
-#define QCA808X_MMD3_CDT_DIAG_PAIR_D 0x8068
|
||||
-#define QCA808X_CDT_DIAG_LENGTH_SAME_SHORT GENMASK(15, 8)
|
||||
-#define QCA808X_CDT_DIAG_LENGTH_CROSS_SHORT GENMASK(7, 0)
|
||||
-
|
||||
-#define QCA808X_CDT_CODE_PAIR_A GENMASK(15, 12)
|
||||
-#define QCA808X_CDT_CODE_PAIR_B GENMASK(11, 8)
|
||||
-#define QCA808X_CDT_CODE_PAIR_C GENMASK(7, 4)
|
||||
-#define QCA808X_CDT_CODE_PAIR_D GENMASK(3, 0)
|
||||
-
|
||||
-#define QCA808X_CDT_STATUS_STAT_TYPE GENMASK(1, 0)
|
||||
-#define QCA808X_CDT_STATUS_STAT_FAIL FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 0)
|
||||
-#define QCA808X_CDT_STATUS_STAT_NORMAL FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 1)
|
||||
-#define QCA808X_CDT_STATUS_STAT_SAME_OPEN FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 2)
|
||||
-#define QCA808X_CDT_STATUS_STAT_SAME_SHORT FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 3)
|
||||
-
|
||||
-#define QCA808X_CDT_STATUS_STAT_MDI GENMASK(3, 2)
|
||||
-#define QCA808X_CDT_STATUS_STAT_MDI1 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 1)
|
||||
-#define QCA808X_CDT_STATUS_STAT_MDI2 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 2)
|
||||
-#define QCA808X_CDT_STATUS_STAT_MDI3 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 3)
|
||||
-
|
||||
-/* NORMAL are MDI with type set to 0 */
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI1
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
- QCA808X_CDT_STATUS_STAT_MDI1)
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
- QCA808X_CDT_STATUS_STAT_MDI1)
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI2
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
- QCA808X_CDT_STATUS_STAT_MDI2)
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
- QCA808X_CDT_STATUS_STAT_MDI2)
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI3
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
- QCA808X_CDT_STATUS_STAT_MDI3)
|
||||
-#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
- QCA808X_CDT_STATUS_STAT_MDI3)
|
||||
-
|
||||
-/* Added for reference of existence but should be handled by wait_for_completion already */
|
||||
-#define QCA808X_CDT_STATUS_STAT_BUSY (BIT(1) | BIT(3))
|
||||
-
|
||||
#define QCA808X_MMD7_LED_GLOBAL 0x8073
|
||||
#define QCA808X_LED_BLINK_1 GENMASK(11, 6)
|
||||
#define QCA808X_LED_BLINK_2 GENMASK(5, 0)
|
||||
@@ -406,86 +356,6 @@ static int qca808x_soft_reset(struct phy
|
||||
return ret;
|
||||
}
|
||||
|
||||
-static bool qca808x_cdt_fault_length_valid(int cdt_code)
|
||||
-{
|
||||
- switch (cdt_code) {
|
||||
- case QCA808X_CDT_STATUS_STAT_SAME_SHORT:
|
||||
- case QCA808X_CDT_STATUS_STAT_SAME_OPEN:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT:
|
||||
- return true;
|
||||
- default:
|
||||
- return false;
|
||||
- }
|
||||
-}
|
||||
-
|
||||
-static int qca808x_cable_test_result_trans(int cdt_code)
|
||||
-{
|
||||
- switch (cdt_code) {
|
||||
- case QCA808X_CDT_STATUS_STAT_NORMAL:
|
||||
- return ETHTOOL_A_CABLE_RESULT_CODE_OK;
|
||||
- case QCA808X_CDT_STATUS_STAT_SAME_SHORT:
|
||||
- return ETHTOOL_A_CABLE_RESULT_CODE_SAME_SHORT;
|
||||
- case QCA808X_CDT_STATUS_STAT_SAME_OPEN:
|
||||
- return ETHTOOL_A_CABLE_RESULT_CODE_OPEN;
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN:
|
||||
- case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT:
|
||||
- return ETHTOOL_A_CABLE_RESULT_CODE_CROSS_SHORT;
|
||||
- case QCA808X_CDT_STATUS_STAT_FAIL:
|
||||
- default:
|
||||
- return ETHTOOL_A_CABLE_RESULT_CODE_UNSPEC;
|
||||
- }
|
||||
-}
|
||||
-
|
||||
-static int qca808x_cdt_fault_length(struct phy_device *phydev, int pair,
|
||||
- int result)
|
||||
-{
|
||||
- int val;
|
||||
- u32 cdt_length_reg = 0;
|
||||
-
|
||||
- switch (pair) {
|
||||
- case ETHTOOL_A_CABLE_PAIR_A:
|
||||
- cdt_length_reg = QCA808X_MMD3_CDT_DIAG_PAIR_A;
|
||||
- break;
|
||||
- case ETHTOOL_A_CABLE_PAIR_B:
|
||||
- cdt_length_reg = QCA808X_MMD3_CDT_DIAG_PAIR_B;
|
||||
- break;
|
||||
- case ETHTOOL_A_CABLE_PAIR_C:
|
||||
- cdt_length_reg = QCA808X_MMD3_CDT_DIAG_PAIR_C;
|
||||
- break;
|
||||
- case ETHTOOL_A_CABLE_PAIR_D:
|
||||
- cdt_length_reg = QCA808X_MMD3_CDT_DIAG_PAIR_D;
|
||||
- break;
|
||||
- default:
|
||||
- return -EINVAL;
|
||||
- }
|
||||
-
|
||||
- val = phy_read_mmd(phydev, MDIO_MMD_PCS, cdt_length_reg);
|
||||
- if (val < 0)
|
||||
- return val;
|
||||
-
|
||||
- if (result == ETHTOOL_A_CABLE_RESULT_CODE_SAME_SHORT)
|
||||
- val = FIELD_GET(QCA808X_CDT_DIAG_LENGTH_SAME_SHORT, val);
|
||||
- else
|
||||
- val = FIELD_GET(QCA808X_CDT_DIAG_LENGTH_CROSS_SHORT, val);
|
||||
-
|
||||
- return at803x_cdt_fault_length(val);
|
||||
-}
|
||||
-
|
||||
static int qca808x_cable_test_start(struct phy_device *phydev)
|
||||
{
|
||||
int ret;
|
||||
@@ -526,81 +396,6 @@ static int qca808x_cable_test_start(stru
|
||||
|
||||
return 0;
|
||||
}
|
||||
-
|
||||
-static int qca808x_cable_test_get_pair_status(struct phy_device *phydev, u8 pair,
|
||||
- u16 status)
|
||||
-{
|
||||
- int length, result;
|
||||
- u16 pair_code;
|
||||
-
|
||||
- switch (pair) {
|
||||
- case ETHTOOL_A_CABLE_PAIR_A:
|
||||
- pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_A, status);
|
||||
- break;
|
||||
- case ETHTOOL_A_CABLE_PAIR_B:
|
||||
- pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_B, status);
|
||||
- break;
|
||||
- case ETHTOOL_A_CABLE_PAIR_C:
|
||||
- pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_C, status);
|
||||
- break;
|
||||
- case ETHTOOL_A_CABLE_PAIR_D:
|
||||
- pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_D, status);
|
||||
- break;
|
||||
- default:
|
||||
- return -EINVAL;
|
||||
- }
|
||||
-
|
||||
- result = qca808x_cable_test_result_trans(pair_code);
|
||||
- ethnl_cable_test_result(phydev, pair, result);
|
||||
-
|
||||
- if (qca808x_cdt_fault_length_valid(pair_code)) {
|
||||
- length = qca808x_cdt_fault_length(phydev, pair, result);
|
||||
- ethnl_cable_test_fault_length(phydev, pair, length);
|
||||
- }
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
-
|
||||
-static int qca808x_cable_test_get_status(struct phy_device *phydev, bool *finished)
|
||||
-{
|
||||
- int ret, val;
|
||||
-
|
||||
- *finished = false;
|
||||
-
|
||||
- val = QCA808X_CDT_ENABLE_TEST |
|
||||
- QCA808X_CDT_LENGTH_UNIT;
|
||||
- ret = at803x_cdt_start(phydev, val);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- ret = at803x_cdt_wait_for_completion(phydev, QCA808X_CDT_ENABLE_TEST);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- val = phy_read_mmd(phydev, MDIO_MMD_PCS, QCA808X_MMD3_CDT_STATUS);
|
||||
- if (val < 0)
|
||||
- return val;
|
||||
-
|
||||
- ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_A, val);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_B, val);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_C, val);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_D, val);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- *finished = true;
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
|
||||
static int qca808x_get_features(struct phy_device *phydev)
|
||||
{
|
||||
--- a/drivers/net/phy/qcom/qcom-phy-lib.c
|
||||
+++ b/drivers/net/phy/qcom/qcom-phy-lib.c
|
||||
@@ -5,6 +5,7 @@
|
||||
|
||||
#include <linux/netdevice.h>
|
||||
#include <linux/etherdevice.h>
|
||||
+#include <linux/ethtool_netlink.h>
|
||||
|
||||
#include "qcom.h"
|
||||
|
||||
@@ -311,6 +312,42 @@ int at803x_prepare_config_aneg(struct ph
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(at803x_prepare_config_aneg);
|
||||
|
||||
+int at803x_read_status(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct at803x_ss_mask ss_mask = { 0 };
|
||||
+ int err, old_link = phydev->link;
|
||||
+
|
||||
+ /* Update the link, but return if there was an error */
|
||||
+ err = genphy_update_link(phydev);
|
||||
+ if (err)
|
||||
+ return err;
|
||||
+
|
||||
+ /* why bother the PHY if nothing can have changed */
|
||||
+ if (phydev->autoneg == AUTONEG_ENABLE && old_link && phydev->link)
|
||||
+ return 0;
|
||||
+
|
||||
+ phydev->speed = SPEED_UNKNOWN;
|
||||
+ phydev->duplex = DUPLEX_UNKNOWN;
|
||||
+ phydev->pause = 0;
|
||||
+ phydev->asym_pause = 0;
|
||||
+
|
||||
+ err = genphy_read_lpa(phydev);
|
||||
+ if (err < 0)
|
||||
+ return err;
|
||||
+
|
||||
+ ss_mask.speed_mask = AT803X_SS_SPEED_MASK;
|
||||
+ ss_mask.speed_shift = __bf_shf(AT803X_SS_SPEED_MASK);
|
||||
+ err = at803x_read_specific_status(phydev, ss_mask);
|
||||
+ if (err < 0)
|
||||
+ return err;
|
||||
+
|
||||
+ if (phydev->autoneg == AUTONEG_ENABLE && phydev->autoneg_complete)
|
||||
+ phy_resolve_aneg_pause(phydev);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(at803x_read_status);
|
||||
+
|
||||
static int at803x_get_downshift(struct phy_device *phydev, u8 *d)
|
||||
{
|
||||
int val;
|
||||
@@ -427,3 +464,159 @@ int at803x_cdt_wait_for_completion(struc
|
||||
return ret < 0 ? ret : 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(at803x_cdt_wait_for_completion);
|
||||
+
|
||||
+static bool qca808x_cdt_fault_length_valid(int cdt_code)
|
||||
+{
|
||||
+ switch (cdt_code) {
|
||||
+ case QCA808X_CDT_STATUS_STAT_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT:
|
||||
+ return true;
|
||||
+ default:
|
||||
+ return false;
|
||||
+ }
|
||||
+}
|
||||
+
|
||||
+static int qca808x_cable_test_result_trans(int cdt_code)
|
||||
+{
|
||||
+ switch (cdt_code) {
|
||||
+ case QCA808X_CDT_STATUS_STAT_NORMAL:
|
||||
+ return ETHTOOL_A_CABLE_RESULT_CODE_OK;
|
||||
+ case QCA808X_CDT_STATUS_STAT_SAME_SHORT:
|
||||
+ return ETHTOOL_A_CABLE_RESULT_CODE_SAME_SHORT;
|
||||
+ case QCA808X_CDT_STATUS_STAT_SAME_OPEN:
|
||||
+ return ETHTOOL_A_CABLE_RESULT_CODE_OPEN;
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN:
|
||||
+ case QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT:
|
||||
+ return ETHTOOL_A_CABLE_RESULT_CODE_CROSS_SHORT;
|
||||
+ case QCA808X_CDT_STATUS_STAT_FAIL:
|
||||
+ default:
|
||||
+ return ETHTOOL_A_CABLE_RESULT_CODE_UNSPEC;
|
||||
+ }
|
||||
+}
|
||||
+
|
||||
+static int qca808x_cdt_fault_length(struct phy_device *phydev, int pair,
|
||||
+ int result)
|
||||
+{
|
||||
+ int val;
|
||||
+ u32 cdt_length_reg = 0;
|
||||
+
|
||||
+ switch (pair) {
|
||||
+ case ETHTOOL_A_CABLE_PAIR_A:
|
||||
+ cdt_length_reg = QCA808X_MMD3_CDT_DIAG_PAIR_A;
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_B:
|
||||
+ cdt_length_reg = QCA808X_MMD3_CDT_DIAG_PAIR_B;
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_C:
|
||||
+ cdt_length_reg = QCA808X_MMD3_CDT_DIAG_PAIR_C;
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_D:
|
||||
+ cdt_length_reg = QCA808X_MMD3_CDT_DIAG_PAIR_D;
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ val = phy_read_mmd(phydev, MDIO_MMD_PCS, cdt_length_reg);
|
||||
+ if (val < 0)
|
||||
+ return val;
|
||||
+
|
||||
+ if (result == ETHTOOL_A_CABLE_RESULT_CODE_SAME_SHORT)
|
||||
+ val = FIELD_GET(QCA808X_CDT_DIAG_LENGTH_SAME_SHORT, val);
|
||||
+ else
|
||||
+ val = FIELD_GET(QCA808X_CDT_DIAG_LENGTH_CROSS_SHORT, val);
|
||||
+
|
||||
+ return at803x_cdt_fault_length(val);
|
||||
+}
|
||||
+
|
||||
+static int qca808x_cable_test_get_pair_status(struct phy_device *phydev, u8 pair,
|
||||
+ u16 status)
|
||||
+{
|
||||
+ int length, result;
|
||||
+ u16 pair_code;
|
||||
+
|
||||
+ switch (pair) {
|
||||
+ case ETHTOOL_A_CABLE_PAIR_A:
|
||||
+ pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_A, status);
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_B:
|
||||
+ pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_B, status);
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_C:
|
||||
+ pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_C, status);
|
||||
+ break;
|
||||
+ case ETHTOOL_A_CABLE_PAIR_D:
|
||||
+ pair_code = FIELD_GET(QCA808X_CDT_CODE_PAIR_D, status);
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ result = qca808x_cable_test_result_trans(pair_code);
|
||||
+ ethnl_cable_test_result(phydev, pair, result);
|
||||
+
|
||||
+ if (qca808x_cdt_fault_length_valid(pair_code)) {
|
||||
+ length = qca808x_cdt_fault_length(phydev, pair, result);
|
||||
+ ethnl_cable_test_fault_length(phydev, pair, length);
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+int qca808x_cable_test_get_status(struct phy_device *phydev, bool *finished)
|
||||
+{
|
||||
+ int ret, val;
|
||||
+
|
||||
+ *finished = false;
|
||||
+
|
||||
+ val = QCA808X_CDT_ENABLE_TEST |
|
||||
+ QCA808X_CDT_LENGTH_UNIT;
|
||||
+ ret = at803x_cdt_start(phydev, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = at803x_cdt_wait_for_completion(phydev, QCA808X_CDT_ENABLE_TEST);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ val = phy_read_mmd(phydev, MDIO_MMD_PCS, QCA808X_MMD3_CDT_STATUS);
|
||||
+ if (val < 0)
|
||||
+ return val;
|
||||
+
|
||||
+ ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_A, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_B, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_C, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = qca808x_cable_test_get_pair_status(phydev, ETHTOOL_A_CABLE_PAIR_D, val);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ *finished = true;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(qca808x_cable_test_get_status);
|
||||
--- a/drivers/net/phy/qcom/qcom.h
|
||||
+++ b/drivers/net/phy/qcom/qcom.h
|
||||
@@ -54,6 +54,55 @@
|
||||
#define AT803X_CDT_STATUS_STAT_MASK GENMASK(9, 8)
|
||||
#define AT803X_CDT_STATUS_DELTA_TIME_MASK GENMASK(7, 0)
|
||||
|
||||
+#define QCA808X_CDT_ENABLE_TEST BIT(15)
|
||||
+#define QCA808X_CDT_INTER_CHECK_DIS BIT(13)
|
||||
+#define QCA808X_CDT_STATUS BIT(11)
|
||||
+#define QCA808X_CDT_LENGTH_UNIT BIT(10)
|
||||
+
|
||||
+#define QCA808X_MMD3_CDT_STATUS 0x8064
|
||||
+#define QCA808X_MMD3_CDT_DIAG_PAIR_A 0x8065
|
||||
+#define QCA808X_MMD3_CDT_DIAG_PAIR_B 0x8066
|
||||
+#define QCA808X_MMD3_CDT_DIAG_PAIR_C 0x8067
|
||||
+#define QCA808X_MMD3_CDT_DIAG_PAIR_D 0x8068
|
||||
+#define QCA808X_CDT_DIAG_LENGTH_SAME_SHORT GENMASK(15, 8)
|
||||
+#define QCA808X_CDT_DIAG_LENGTH_CROSS_SHORT GENMASK(7, 0)
|
||||
+
|
||||
+#define QCA808X_CDT_CODE_PAIR_A GENMASK(15, 12)
|
||||
+#define QCA808X_CDT_CODE_PAIR_B GENMASK(11, 8)
|
||||
+#define QCA808X_CDT_CODE_PAIR_C GENMASK(7, 4)
|
||||
+#define QCA808X_CDT_CODE_PAIR_D GENMASK(3, 0)
|
||||
+
|
||||
+#define QCA808X_CDT_STATUS_STAT_TYPE GENMASK(1, 0)
|
||||
+#define QCA808X_CDT_STATUS_STAT_FAIL FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 0)
|
||||
+#define QCA808X_CDT_STATUS_STAT_NORMAL FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 1)
|
||||
+#define QCA808X_CDT_STATUS_STAT_SAME_OPEN FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_SAME_SHORT FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_TYPE, 3)
|
||||
+
|
||||
+#define QCA808X_CDT_STATUS_STAT_MDI GENMASK(3, 2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_MDI1 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 1)
|
||||
+#define QCA808X_CDT_STATUS_STAT_MDI2 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_MDI3 FIELD_PREP_CONST(QCA808X_CDT_STATUS_STAT_MDI, 3)
|
||||
+
|
||||
+/* NORMAL are MDI with type set to 0 */
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI1
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI1)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI1_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI1)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI2
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI2_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI2)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_NORMAL QCA808X_CDT_STATUS_STAT_MDI3
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_OPEN (QCA808X_CDT_STATUS_STAT_SAME_OPEN |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI3)
|
||||
+#define QCA808X_CDT_STATUS_STAT_CROSS_SHORT_WITH_MDI3_SAME_SHORT (QCA808X_CDT_STATUS_STAT_SAME_SHORT |\
|
||||
+ QCA808X_CDT_STATUS_STAT_MDI3)
|
||||
+
|
||||
+/* Added for reference of existence but should be handled by wait_for_completion already */
|
||||
+#define QCA808X_CDT_STATUS_STAT_BUSY (BIT(1) | BIT(3))
|
||||
+
|
||||
#define AT803X_LOC_MAC_ADDR_0_15_OFFSET 0x804C
|
||||
#define AT803X_LOC_MAC_ADDR_16_31_OFFSET 0x804B
|
||||
#define AT803X_LOC_MAC_ADDR_32_47_OFFSET 0x804A
|
||||
@@ -110,6 +159,7 @@ int at803x_read_specific_status(struct p
|
||||
struct at803x_ss_mask ss_mask);
|
||||
int at803x_config_mdix(struct phy_device *phydev, u8 ctrl);
|
||||
int at803x_prepare_config_aneg(struct phy_device *phydev);
|
||||
+int at803x_read_status(struct phy_device *phydev);
|
||||
int at803x_get_tunable(struct phy_device *phydev,
|
||||
struct ethtool_tunable *tuna, void *data);
|
||||
int at803x_set_tunable(struct phy_device *phydev,
|
||||
@@ -118,3 +168,4 @@ int at803x_cdt_fault_length(int dt);
|
||||
int at803x_cdt_start(struct phy_device *phydev, u32 cdt_start);
|
||||
int at803x_cdt_wait_for_completion(struct phy_device *phydev,
|
||||
u32 cdt_en);
|
||||
+int qca808x_cable_test_get_status(struct phy_device *phydev, bool *finished);
|
@ -0,0 +1,100 @@
|
||||
From 9b1d5e055508393561e26bd1720f4c2639b03b1a Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 6 Feb 2024 18:31:09 +0100
|
||||
Subject: [PATCH 06/10] net: phy: provide whether link has changed in
|
||||
c37_read_status
|
||||
|
||||
Some PHY driver might require additional regs call after
|
||||
genphy_c37_read_status() is called.
|
||||
|
||||
Expand genphy_c37_read_status to provide a bool wheather the link has
|
||||
changed or not to permit PHY driver to skip additional regs call if
|
||||
nothing has changed.
|
||||
|
||||
Every user of genphy_c37_read_status() is updated with the new
|
||||
additional bool.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/broadcom.c | 3 ++-
|
||||
drivers/net/phy/phy_device.c | 11 +++++++++--
|
||||
drivers/net/phy/qcom/at803x.c | 3 ++-
|
||||
include/linux/phy.h | 2 +-
|
||||
4 files changed, 14 insertions(+), 5 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/broadcom.c
|
||||
+++ b/drivers/net/phy/broadcom.c
|
||||
@@ -665,10 +665,11 @@ static int bcm54616s_config_aneg(struct
|
||||
static int bcm54616s_read_status(struct phy_device *phydev)
|
||||
{
|
||||
struct bcm54616s_phy_priv *priv = phydev->priv;
|
||||
+ bool changed;
|
||||
int err;
|
||||
|
||||
if (priv->mode_1000bx_en)
|
||||
- err = genphy_c37_read_status(phydev);
|
||||
+ err = genphy_c37_read_status(phydev, &changed);
|
||||
else
|
||||
err = genphy_read_status(phydev);
|
||||
|
||||
--- a/drivers/net/phy/phy_device.c
|
||||
+++ b/drivers/net/phy/phy_device.c
|
||||
@@ -2607,12 +2607,15 @@ EXPORT_SYMBOL(genphy_read_status);
|
||||
/**
|
||||
* genphy_c37_read_status - check the link status and update current link state
|
||||
* @phydev: target phy_device struct
|
||||
+ * @changed: pointer where to store if link changed
|
||||
*
|
||||
* Description: Check the link, then figure out the current state
|
||||
* by comparing what we advertise with what the link partner
|
||||
* advertises. This function is for Clause 37 1000Base-X mode.
|
||||
+ *
|
||||
+ * If link has changed, @changed is set to true, false otherwise.
|
||||
*/
|
||||
-int genphy_c37_read_status(struct phy_device *phydev)
|
||||
+int genphy_c37_read_status(struct phy_device *phydev, bool *changed)
|
||||
{
|
||||
int lpa, err, old_link = phydev->link;
|
||||
|
||||
@@ -2622,9 +2625,13 @@ int genphy_c37_read_status(struct phy_de
|
||||
return err;
|
||||
|
||||
/* why bother the PHY if nothing can have changed */
|
||||
- if (phydev->autoneg == AUTONEG_ENABLE && old_link && phydev->link)
|
||||
+ if (phydev->autoneg == AUTONEG_ENABLE && old_link && phydev->link) {
|
||||
+ *changed = false;
|
||||
return 0;
|
||||
+ }
|
||||
|
||||
+ /* Signal link has changed */
|
||||
+ *changed = true;
|
||||
phydev->duplex = DUPLEX_UNKNOWN;
|
||||
phydev->pause = 0;
|
||||
phydev->asym_pause = 0;
|
||||
--- a/drivers/net/phy/qcom/at803x.c
|
||||
+++ b/drivers/net/phy/qcom/at803x.c
|
||||
@@ -912,9 +912,10 @@ static int at8031_config_intr(struct phy
|
||||
static int at8031_read_status(struct phy_device *phydev)
|
||||
{
|
||||
struct at803x_priv *priv = phydev->priv;
|
||||
+ bool changed;
|
||||
|
||||
if (priv->is_1000basex)
|
||||
- return genphy_c37_read_status(phydev);
|
||||
+ return genphy_c37_read_status(phydev, &changed);
|
||||
|
||||
return at803x_read_status(phydev);
|
||||
}
|
||||
--- a/include/linux/phy.h
|
||||
+++ b/include/linux/phy.h
|
||||
@@ -1849,7 +1849,7 @@ int genphy_write_mmd_unsupported(struct
|
||||
|
||||
/* Clause 37 */
|
||||
int genphy_c37_config_aneg(struct phy_device *phydev);
|
||||
-int genphy_c37_read_status(struct phy_device *phydev);
|
||||
+int genphy_c37_read_status(struct phy_device *phydev, bool *changed);
|
||||
|
||||
/* Clause 45 PHY */
|
||||
int genphy_c45_restart_aneg(struct phy_device *phydev);
|
@ -0,0 +1,668 @@
|
||||
From d1cb613efbd3cd7d0c000167816beb3f248f5eb8 Mon Sep 17 00:00:00 2001
|
||||
From: Robert Marko <robert.marko@sartura.hr>
|
||||
Date: Tue, 6 Feb 2024 18:31:10 +0100
|
||||
Subject: [PATCH 07/10] net: phy: qcom: add support for QCA807x PHY Family
|
||||
|
||||
This adds driver for the Qualcomm QCA8072 and QCA8075 PHY-s.
|
||||
|
||||
They are 2 or 5 port IEEE 802.3 clause 22 compliant 10BASE-Te,
|
||||
100BASE-TX and 1000BASE-T PHY-s.
|
||||
|
||||
They feature 2 SerDes, one for PSGMII or QSGMII connection with
|
||||
MAC, while second one is SGMII for connection to MAC or fiber.
|
||||
|
||||
Both models have a combo port that supports 1000BASE-X and
|
||||
100BASE-FX fiber.
|
||||
|
||||
PHY package can be configured in 3 mode following this table:
|
||||
|
||||
First Serdes mode Second Serdes mode
|
||||
Option 1 PSGMII for copper Disabled
|
||||
ports 0-4
|
||||
Option 2 PSGMII for copper 1000BASE-X / 100BASE-FX
|
||||
ports 0-4
|
||||
Option 3 QSGMII for copper SGMII for
|
||||
ports 0-3 copper port 4
|
||||
|
||||
Each PHY inside of QCA807x series has 4 digitally controlled
|
||||
output only pins that natively drive LED-s.
|
||||
But some vendors used these to driver generic LED-s controlled
|
||||
by userspace, so lets enable registering each PHY as GPIO
|
||||
controller and add driver for it.
|
||||
|
||||
These are commonly used in Qualcomm IPQ40xx, IPQ60xx and IPQ807x
|
||||
boards.
|
||||
|
||||
Co-developed-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: Robert Marko <robert.marko@sartura.hr>
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/qcom/Kconfig | 8 +
|
||||
drivers/net/phy/qcom/Makefile | 1 +
|
||||
drivers/net/phy/qcom/qca807x.c | 597 +++++++++++++++++++++++++++++++++
|
||||
3 files changed, 606 insertions(+)
|
||||
create mode 100644 drivers/net/phy/qcom/qca807x.c
|
||||
|
||||
--- a/drivers/net/phy/qcom/Kconfig
|
||||
+++ b/drivers/net/phy/qcom/Kconfig
|
||||
@@ -20,3 +20,11 @@ config QCA808X_PHY
|
||||
select QCOM_NET_PHYLIB
|
||||
help
|
||||
Currently supports the QCA8081 model
|
||||
+
|
||||
+config QCA807X_PHY
|
||||
+ tristate "Qualcomm QCA807x PHYs"
|
||||
+ select QCOM_NET_PHYLIB
|
||||
+ depends on OF_MDIO
|
||||
+ help
|
||||
+ Currently supports the Qualcomm QCA8072, QCA8075 and the PSGMII
|
||||
+ control PHY.
|
||||
--- a/drivers/net/phy/qcom/Makefile
|
||||
+++ b/drivers/net/phy/qcom/Makefile
|
||||
@@ -3,3 +3,4 @@ obj-$(CONFIG_QCOM_NET_PHYLIB) += qcom-ph
|
||||
obj-$(CONFIG_AT803X_PHY) += at803x.o
|
||||
obj-$(CONFIG_QCA83XX_PHY) += qca83xx.o
|
||||
obj-$(CONFIG_QCA808X_PHY) += qca808x.o
|
||||
+obj-$(CONFIG_QCA807X_PHY) += qca807x.o
|
||||
--- /dev/null
|
||||
+++ b/drivers/net/phy/qcom/qca807x.c
|
||||
@@ -0,0 +1,597 @@
|
||||
+// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
+/*
|
||||
+ * Copyright (c) 2023 Sartura Ltd.
|
||||
+ *
|
||||
+ * Author: Robert Marko <robert.marko@sartura.hr>
|
||||
+ * Christian Marangi <ansuelsmth@gmail.com>
|
||||
+ *
|
||||
+ * Qualcomm QCA8072 and QCA8075 PHY driver
|
||||
+ */
|
||||
+
|
||||
+#include <linux/module.h>
|
||||
+#include <linux/of.h>
|
||||
+#include <linux/phy.h>
|
||||
+#include <linux/bitfield.h>
|
||||
+#include <linux/gpio/driver.h>
|
||||
+#include <linux/sfp.h>
|
||||
+
|
||||
+#include "qcom.h"
|
||||
+
|
||||
+#define QCA807X_CHIP_CONFIGURATION 0x1f
|
||||
+#define QCA807X_BT_BX_REG_SEL BIT(15)
|
||||
+#define QCA807X_BT_BX_REG_SEL_FIBER 0
|
||||
+#define QCA807X_BT_BX_REG_SEL_COPPER 1
|
||||
+#define QCA807X_CHIP_CONFIGURATION_MODE_CFG_MASK GENMASK(3, 0)
|
||||
+#define QCA807X_CHIP_CONFIGURATION_MODE_QSGMII_SGMII 4
|
||||
+#define QCA807X_CHIP_CONFIGURATION_MODE_PSGMII_FIBER 3
|
||||
+#define QCA807X_CHIP_CONFIGURATION_MODE_PSGMII_ALL_COPPER 0
|
||||
+
|
||||
+#define QCA807X_MEDIA_SELECT_STATUS 0x1a
|
||||
+#define QCA807X_MEDIA_DETECTED_COPPER BIT(5)
|
||||
+#define QCA807X_MEDIA_DETECTED_1000_BASE_X BIT(4)
|
||||
+#define QCA807X_MEDIA_DETECTED_100_BASE_FX BIT(3)
|
||||
+
|
||||
+#define QCA807X_MMD7_FIBER_MODE_AUTO_DETECTION 0x807e
|
||||
+#define QCA807X_MMD7_FIBER_MODE_AUTO_DETECTION_EN BIT(0)
|
||||
+
|
||||
+#define QCA807X_MMD7_1000BASE_T_POWER_SAVE_PER_CABLE_LENGTH 0x801a
|
||||
+#define QCA807X_CONTROL_DAC_MASK GENMASK(2, 0)
|
||||
+/* List of tweaks enabled by this bit:
|
||||
+ * - With both FULL amplitude and FULL bias current: bias current
|
||||
+ * is set to half.
|
||||
+ * - With only DSP amplitude: bias current is set to half and
|
||||
+ * is set to 1/4 with cable < 10m.
|
||||
+ * - With DSP bias current (included both DSP amplitude and
|
||||
+ * DSP bias current): bias current is half the detected current
|
||||
+ * with cable < 10m.
|
||||
+ */
|
||||
+#define QCA807X_CONTROL_DAC_BIAS_CURRENT_TWEAK BIT(2)
|
||||
+#define QCA807X_CONTROL_DAC_DSP_BIAS_CURRENT BIT(1)
|
||||
+#define QCA807X_CONTROL_DAC_DSP_AMPLITUDE BIT(0)
|
||||
+
|
||||
+#define QCA807X_MMD7_LED_100N_1 0x8074
|
||||
+#define QCA807X_MMD7_LED_100N_2 0x8075
|
||||
+#define QCA807X_MMD7_LED_1000N_1 0x8076
|
||||
+#define QCA807X_MMD7_LED_1000N_2 0x8077
|
||||
+
|
||||
+#define QCA807X_MMD7_LED_CTRL(x) (0x8074 + ((x) * 2))
|
||||
+#define QCA807X_MMD7_LED_FORCE_CTRL(x) (0x8075 + ((x) * 2))
|
||||
+
|
||||
+#define QCA807X_GPIO_FORCE_EN BIT(15)
|
||||
+#define QCA807X_GPIO_FORCE_MODE_MASK GENMASK(14, 13)
|
||||
+
|
||||
+#define QCA807X_FUNCTION_CONTROL 0x10
|
||||
+#define QCA807X_FC_MDI_CROSSOVER_MODE_MASK GENMASK(6, 5)
|
||||
+#define QCA807X_FC_MDI_CROSSOVER_AUTO 3
|
||||
+#define QCA807X_FC_MDI_CROSSOVER_MANUAL_MDIX 1
|
||||
+#define QCA807X_FC_MDI_CROSSOVER_MANUAL_MDI 0
|
||||
+
|
||||
+/* PQSGMII Analog PHY specific */
|
||||
+#define PQSGMII_CTRL_REG 0x0
|
||||
+#define PQSGMII_ANALOG_SW_RESET BIT(6)
|
||||
+#define PQSGMII_DRIVE_CONTROL_1 0xb
|
||||
+#define PQSGMII_TX_DRIVER_MASK GENMASK(7, 4)
|
||||
+#define PQSGMII_TX_DRIVER_140MV 0x0
|
||||
+#define PQSGMII_TX_DRIVER_160MV 0x1
|
||||
+#define PQSGMII_TX_DRIVER_180MV 0x2
|
||||
+#define PQSGMII_TX_DRIVER_200MV 0x3
|
||||
+#define PQSGMII_TX_DRIVER_220MV 0x4
|
||||
+#define PQSGMII_TX_DRIVER_240MV 0x5
|
||||
+#define PQSGMII_TX_DRIVER_260MV 0x6
|
||||
+#define PQSGMII_TX_DRIVER_280MV 0x7
|
||||
+#define PQSGMII_TX_DRIVER_300MV 0x8
|
||||
+#define PQSGMII_TX_DRIVER_320MV 0x9
|
||||
+#define PQSGMII_TX_DRIVER_400MV 0xa
|
||||
+#define PQSGMII_TX_DRIVER_500MV 0xb
|
||||
+#define PQSGMII_TX_DRIVER_600MV 0xc
|
||||
+#define PQSGMII_MODE_CTRL 0x6d
|
||||
+#define PQSGMII_MODE_CTRL_AZ_WORKAROUND_MASK BIT(0)
|
||||
+#define PQSGMII_MMD3_SERDES_CONTROL 0x805a
|
||||
+
|
||||
+#define PHY_ID_QCA8072 0x004dd0b2
|
||||
+#define PHY_ID_QCA8075 0x004dd0b1
|
||||
+
|
||||
+#define QCA807X_COMBO_ADDR_OFFSET 4
|
||||
+#define QCA807X_PQSGMII_ADDR_OFFSET 5
|
||||
+#define SERDES_RESET_SLEEP 100
|
||||
+
|
||||
+enum qca807x_global_phy {
|
||||
+ QCA807X_COMBO_ADDR = 4,
|
||||
+ QCA807X_PQSGMII_ADDR = 5,
|
||||
+};
|
||||
+
|
||||
+struct qca807x_shared_priv {
|
||||
+ unsigned int package_mode;
|
||||
+ u32 tx_drive_strength;
|
||||
+};
|
||||
+
|
||||
+struct qca807x_gpio_priv {
|
||||
+ struct phy_device *phy;
|
||||
+};
|
||||
+
|
||||
+struct qca807x_priv {
|
||||
+ bool dac_full_amplitude;
|
||||
+ bool dac_full_bias_current;
|
||||
+ bool dac_disable_bias_current_tweak;
|
||||
+};
|
||||
+
|
||||
+static int qca807x_cable_test_start(struct phy_device *phydev)
|
||||
+{
|
||||
+ /* we do all the (time consuming) work later */
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+#ifdef CONFIG_GPIOLIB
|
||||
+static int qca807x_gpio_get_direction(struct gpio_chip *gc, unsigned int offset)
|
||||
+{
|
||||
+ return GPIO_LINE_DIRECTION_OUT;
|
||||
+}
|
||||
+
|
||||
+static int qca807x_gpio_get(struct gpio_chip *gc, unsigned int offset)
|
||||
+{
|
||||
+ struct qca807x_gpio_priv *priv = gpiochip_get_data(gc);
|
||||
+ u16 reg;
|
||||
+ int val;
|
||||
+
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(offset);
|
||||
+ val = phy_read_mmd(priv->phy, MDIO_MMD_AN, reg);
|
||||
+
|
||||
+ return FIELD_GET(QCA807X_GPIO_FORCE_MODE_MASK, val);
|
||||
+}
|
||||
+
|
||||
+static void qca807x_gpio_set(struct gpio_chip *gc, unsigned int offset, int value)
|
||||
+{
|
||||
+ struct qca807x_gpio_priv *priv = gpiochip_get_data(gc);
|
||||
+ u16 reg;
|
||||
+ int val;
|
||||
+
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(offset);
|
||||
+
|
||||
+ val = phy_read_mmd(priv->phy, MDIO_MMD_AN, reg);
|
||||
+ val &= ~QCA807X_GPIO_FORCE_MODE_MASK;
|
||||
+ val |= QCA807X_GPIO_FORCE_EN;
|
||||
+ val |= FIELD_PREP(QCA807X_GPIO_FORCE_MODE_MASK, value);
|
||||
+
|
||||
+ phy_write_mmd(priv->phy, MDIO_MMD_AN, reg, val);
|
||||
+}
|
||||
+
|
||||
+static int qca807x_gpio_dir_out(struct gpio_chip *gc, unsigned int offset, int value)
|
||||
+{
|
||||
+ qca807x_gpio_set(gc, offset, value);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca807x_gpio(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct device *dev = &phydev->mdio.dev;
|
||||
+ struct qca807x_gpio_priv *priv;
|
||||
+ struct gpio_chip *gc;
|
||||
+
|
||||
+ priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
|
||||
+ if (!priv)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ priv->phy = phydev;
|
||||
+
|
||||
+ gc = devm_kzalloc(dev, sizeof(*gc), GFP_KERNEL);
|
||||
+ if (!gc)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ gc->label = dev_name(dev);
|
||||
+ gc->base = -1;
|
||||
+ gc->ngpio = 2;
|
||||
+ gc->parent = dev;
|
||||
+ gc->owner = THIS_MODULE;
|
||||
+ gc->can_sleep = true;
|
||||
+ gc->get_direction = qca807x_gpio_get_direction;
|
||||
+ gc->direction_output = qca807x_gpio_dir_out;
|
||||
+ gc->get = qca807x_gpio_get;
|
||||
+ gc->set = qca807x_gpio_set;
|
||||
+
|
||||
+ return devm_gpiochip_add_data(dev, gc, priv);
|
||||
+}
|
||||
+#endif
|
||||
+
|
||||
+static int qca807x_read_fiber_status(struct phy_device *phydev)
|
||||
+{
|
||||
+ bool changed;
|
||||
+ int ss, err;
|
||||
+
|
||||
+ err = genphy_c37_read_status(phydev, &changed);
|
||||
+ if (err || !changed)
|
||||
+ return err;
|
||||
+
|
||||
+ /* Read the QCA807x PHY-Specific Status register fiber page,
|
||||
+ * which indicates the speed and duplex that the PHY is actually
|
||||
+ * using, irrespective of whether we are in autoneg mode or not.
|
||||
+ */
|
||||
+ ss = phy_read(phydev, AT803X_SPECIFIC_STATUS);
|
||||
+ if (ss < 0)
|
||||
+ return ss;
|
||||
+
|
||||
+ phydev->speed = SPEED_UNKNOWN;
|
||||
+ phydev->duplex = DUPLEX_UNKNOWN;
|
||||
+ if (ss & AT803X_SS_SPEED_DUPLEX_RESOLVED) {
|
||||
+ switch (FIELD_GET(AT803X_SS_SPEED_MASK, ss)) {
|
||||
+ case AT803X_SS_SPEED_100:
|
||||
+ phydev->speed = SPEED_100;
|
||||
+ break;
|
||||
+ case AT803X_SS_SPEED_1000:
|
||||
+ phydev->speed = SPEED_1000;
|
||||
+ break;
|
||||
+ }
|
||||
+
|
||||
+ if (ss & AT803X_SS_DUPLEX)
|
||||
+ phydev->duplex = DUPLEX_FULL;
|
||||
+ else
|
||||
+ phydev->duplex = DUPLEX_HALF;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca807x_read_status(struct phy_device *phydev)
|
||||
+{
|
||||
+ if (linkmode_test_bit(ETHTOOL_LINK_MODE_FIBRE_BIT, phydev->supported)) {
|
||||
+ switch (phydev->port) {
|
||||
+ case PORT_FIBRE:
|
||||
+ return qca807x_read_fiber_status(phydev);
|
||||
+ case PORT_TP:
|
||||
+ return at803x_read_status(phydev);
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ return at803x_read_status(phydev);
|
||||
+}
|
||||
+
|
||||
+static int qca807x_phy_package_probe_once(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct phy_package_shared *shared = phydev->shared;
|
||||
+ struct qca807x_shared_priv *priv = shared->priv;
|
||||
+ unsigned int tx_drive_strength;
|
||||
+ const char *package_mode_name;
|
||||
+
|
||||
+ /* Default to 600mw if not defined */
|
||||
+ if (of_property_read_u32(shared->np, "qcom,tx-drive-strength-milliwatt",
|
||||
+ &tx_drive_strength))
|
||||
+ tx_drive_strength = 600;
|
||||
+
|
||||
+ switch (tx_drive_strength) {
|
||||
+ case 140:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_140MV;
|
||||
+ break;
|
||||
+ case 160:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_160MV;
|
||||
+ break;
|
||||
+ case 180:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_180MV;
|
||||
+ break;
|
||||
+ case 200:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_200MV;
|
||||
+ break;
|
||||
+ case 220:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_220MV;
|
||||
+ break;
|
||||
+ case 240:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_240MV;
|
||||
+ break;
|
||||
+ case 260:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_260MV;
|
||||
+ break;
|
||||
+ case 280:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_280MV;
|
||||
+ break;
|
||||
+ case 300:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_300MV;
|
||||
+ break;
|
||||
+ case 320:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_320MV;
|
||||
+ break;
|
||||
+ case 400:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_400MV;
|
||||
+ break;
|
||||
+ case 500:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_500MV;
|
||||
+ break;
|
||||
+ case 600:
|
||||
+ priv->tx_drive_strength = PQSGMII_TX_DRIVER_600MV;
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ priv->package_mode = PHY_INTERFACE_MODE_NA;
|
||||
+ if (!of_property_read_string(shared->np, "qcom,package-mode",
|
||||
+ &package_mode_name)) {
|
||||
+ if (!strcasecmp(package_mode_name,
|
||||
+ phy_modes(PHY_INTERFACE_MODE_PSGMII)))
|
||||
+ priv->package_mode = PHY_INTERFACE_MODE_PSGMII;
|
||||
+ else if (!strcasecmp(package_mode_name,
|
||||
+ phy_modes(PHY_INTERFACE_MODE_QSGMII)))
|
||||
+ priv->package_mode = PHY_INTERFACE_MODE_QSGMII;
|
||||
+ else
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca807x_phy_package_config_init_once(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct phy_package_shared *shared = phydev->shared;
|
||||
+ struct qca807x_shared_priv *priv = shared->priv;
|
||||
+ int val, ret;
|
||||
+
|
||||
+ phy_lock_mdio_bus(phydev);
|
||||
+
|
||||
+ /* Set correct PHY package mode */
|
||||
+ val = __phy_package_read(phydev, QCA807X_COMBO_ADDR,
|
||||
+ QCA807X_CHIP_CONFIGURATION);
|
||||
+ val &= ~QCA807X_CHIP_CONFIGURATION_MODE_CFG_MASK;
|
||||
+ /* package_mode can be QSGMII or PSGMII and we validate
|
||||
+ * this in probe_once.
|
||||
+ * With package_mode to NA, we default to PSGMII.
|
||||
+ */
|
||||
+ switch (priv->package_mode) {
|
||||
+ case PHY_INTERFACE_MODE_QSGMII:
|
||||
+ val |= QCA807X_CHIP_CONFIGURATION_MODE_QSGMII_SGMII;
|
||||
+ break;
|
||||
+ case PHY_INTERFACE_MODE_PSGMII:
|
||||
+ default:
|
||||
+ val |= QCA807X_CHIP_CONFIGURATION_MODE_PSGMII_ALL_COPPER;
|
||||
+ }
|
||||
+ ret = __phy_package_write(phydev, QCA807X_COMBO_ADDR,
|
||||
+ QCA807X_CHIP_CONFIGURATION, val);
|
||||
+ if (ret)
|
||||
+ goto exit;
|
||||
+
|
||||
+ /* After mode change Serdes reset is required */
|
||||
+ val = __phy_package_read(phydev, QCA807X_PQSGMII_ADDR,
|
||||
+ PQSGMII_CTRL_REG);
|
||||
+ val &= ~PQSGMII_ANALOG_SW_RESET;
|
||||
+ ret = __phy_package_write(phydev, QCA807X_PQSGMII_ADDR,
|
||||
+ PQSGMII_CTRL_REG, val);
|
||||
+ if (ret)
|
||||
+ goto exit;
|
||||
+
|
||||
+ msleep(SERDES_RESET_SLEEP);
|
||||
+
|
||||
+ val = __phy_package_read(phydev, QCA807X_PQSGMII_ADDR,
|
||||
+ PQSGMII_CTRL_REG);
|
||||
+ val |= PQSGMII_ANALOG_SW_RESET;
|
||||
+ ret = __phy_package_write(phydev, QCA807X_PQSGMII_ADDR,
|
||||
+ PQSGMII_CTRL_REG, val);
|
||||
+ if (ret)
|
||||
+ goto exit;
|
||||
+
|
||||
+ /* Workaround to enable AZ transmitting ability */
|
||||
+ val = __phy_package_read_mmd(phydev, QCA807X_PQSGMII_ADDR,
|
||||
+ MDIO_MMD_PMAPMD, PQSGMII_MODE_CTRL);
|
||||
+ val &= ~PQSGMII_MODE_CTRL_AZ_WORKAROUND_MASK;
|
||||
+ ret = __phy_package_write_mmd(phydev, QCA807X_PQSGMII_ADDR,
|
||||
+ MDIO_MMD_PMAPMD, PQSGMII_MODE_CTRL, val);
|
||||
+ if (ret)
|
||||
+ goto exit;
|
||||
+
|
||||
+ /* Set PQSGMII TX AMP strength */
|
||||
+ val = __phy_package_read(phydev, QCA807X_PQSGMII_ADDR,
|
||||
+ PQSGMII_DRIVE_CONTROL_1);
|
||||
+ val &= ~PQSGMII_TX_DRIVER_MASK;
|
||||
+ val |= FIELD_PREP(PQSGMII_TX_DRIVER_MASK, priv->tx_drive_strength);
|
||||
+ ret = __phy_package_write(phydev, QCA807X_PQSGMII_ADDR,
|
||||
+ PQSGMII_DRIVE_CONTROL_1, val);
|
||||
+ if (ret)
|
||||
+ goto exit;
|
||||
+
|
||||
+ /* Prevent PSGMII going into hibernation via PSGMII self test */
|
||||
+ val = __phy_package_read_mmd(phydev, QCA807X_COMBO_ADDR,
|
||||
+ MDIO_MMD_PCS, PQSGMII_MMD3_SERDES_CONTROL);
|
||||
+ val &= ~BIT(1);
|
||||
+ ret = __phy_package_write_mmd(phydev, QCA807X_COMBO_ADDR,
|
||||
+ MDIO_MMD_PCS, PQSGMII_MMD3_SERDES_CONTROL, val);
|
||||
+
|
||||
+exit:
|
||||
+ phy_unlock_mdio_bus(phydev);
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
+static int qca807x_sfp_insert(void *upstream, const struct sfp_eeprom_id *id)
|
||||
+{
|
||||
+ struct phy_device *phydev = upstream;
|
||||
+ __ETHTOOL_DECLARE_LINK_MODE_MASK(support) = { 0, };
|
||||
+ phy_interface_t iface;
|
||||
+ int ret;
|
||||
+ DECLARE_PHY_INTERFACE_MASK(interfaces);
|
||||
+
|
||||
+ sfp_parse_support(phydev->sfp_bus, id, support, interfaces);
|
||||
+ iface = sfp_select_interface(phydev->sfp_bus, support);
|
||||
+
|
||||
+ dev_info(&phydev->mdio.dev, "%s SFP module inserted\n", phy_modes(iface));
|
||||
+
|
||||
+ switch (iface) {
|
||||
+ case PHY_INTERFACE_MODE_1000BASEX:
|
||||
+ case PHY_INTERFACE_MODE_100BASEX:
|
||||
+ /* Set PHY mode to PSGMII combo (1/4 copper + combo ports) mode */
|
||||
+ ret = phy_modify(phydev,
|
||||
+ QCA807X_CHIP_CONFIGURATION,
|
||||
+ QCA807X_CHIP_CONFIGURATION_MODE_CFG_MASK,
|
||||
+ QCA807X_CHIP_CONFIGURATION_MODE_PSGMII_FIBER);
|
||||
+ /* Enable fiber mode autodection (1000Base-X or 100Base-FX) */
|
||||
+ ret = phy_set_bits_mmd(phydev,
|
||||
+ MDIO_MMD_AN,
|
||||
+ QCA807X_MMD7_FIBER_MODE_AUTO_DETECTION,
|
||||
+ QCA807X_MMD7_FIBER_MODE_AUTO_DETECTION_EN);
|
||||
+ /* Select fiber page */
|
||||
+ ret = phy_clear_bits(phydev,
|
||||
+ QCA807X_CHIP_CONFIGURATION,
|
||||
+ QCA807X_BT_BX_REG_SEL);
|
||||
+
|
||||
+ phydev->port = PORT_FIBRE;
|
||||
+ break;
|
||||
+ default:
|
||||
+ dev_err(&phydev->mdio.dev, "Incompatible SFP module inserted\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
+static void qca807x_sfp_remove(void *upstream)
|
||||
+{
|
||||
+ struct phy_device *phydev = upstream;
|
||||
+
|
||||
+ /* Select copper page */
|
||||
+ phy_set_bits(phydev,
|
||||
+ QCA807X_CHIP_CONFIGURATION,
|
||||
+ QCA807X_BT_BX_REG_SEL);
|
||||
+
|
||||
+ phydev->port = PORT_TP;
|
||||
+}
|
||||
+
|
||||
+static const struct sfp_upstream_ops qca807x_sfp_ops = {
|
||||
+ .attach = phy_sfp_attach,
|
||||
+ .detach = phy_sfp_detach,
|
||||
+ .module_insert = qca807x_sfp_insert,
|
||||
+ .module_remove = qca807x_sfp_remove,
|
||||
+};
|
||||
+
|
||||
+static int qca807x_probe(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct device_node *node = phydev->mdio.dev.of_node;
|
||||
+ struct qca807x_shared_priv *shared_priv;
|
||||
+ struct device *dev = &phydev->mdio.dev;
|
||||
+ struct phy_package_shared *shared;
|
||||
+ struct qca807x_priv *priv;
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = devm_of_phy_package_join(dev, phydev, sizeof(*shared_priv));
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ if (phy_package_probe_once(phydev)) {
|
||||
+ ret = qca807x_phy_package_probe_once(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ shared = phydev->shared;
|
||||
+ shared_priv = shared->priv;
|
||||
+
|
||||
+ /* Make sure PHY follow PHY package mode if enforced */
|
||||
+ if (shared_priv->package_mode != PHY_INTERFACE_MODE_NA &&
|
||||
+ phydev->interface != shared_priv->package_mode)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
|
||||
+ if (!priv)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ priv->dac_full_amplitude = of_property_read_bool(node, "qcom,dac-full-amplitude");
|
||||
+ priv->dac_full_bias_current = of_property_read_bool(node, "qcom,dac-full-bias-current");
|
||||
+ priv->dac_disable_bias_current_tweak = of_property_read_bool(node,
|
||||
+ "qcom,dac-disable-bias-current-tweak");
|
||||
+
|
||||
+ if (IS_ENABLED(CONFIG_GPIOLIB)) {
|
||||
+ /* Do not register a GPIO controller unless flagged for it */
|
||||
+ if (of_property_read_bool(node, "gpio-controller")) {
|
||||
+ ret = qca807x_gpio(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ /* Attach SFP bus on combo port*/
|
||||
+ if (phy_read(phydev, QCA807X_CHIP_CONFIGURATION)) {
|
||||
+ ret = phy_sfp_probe(phydev, &qca807x_sfp_ops);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+ linkmode_set_bit(ETHTOOL_LINK_MODE_FIBRE_BIT, phydev->supported);
|
||||
+ linkmode_set_bit(ETHTOOL_LINK_MODE_FIBRE_BIT, phydev->advertising);
|
||||
+ }
|
||||
+
|
||||
+ phydev->priv = priv;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca807x_config_init(struct phy_device *phydev)
|
||||
+{
|
||||
+ struct qca807x_priv *priv = phydev->priv;
|
||||
+ u16 control_dac;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (phy_package_init_once(phydev)) {
|
||||
+ ret = qca807x_phy_package_config_init_once(phydev);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ control_dac = phy_read_mmd(phydev, MDIO_MMD_AN,
|
||||
+ QCA807X_MMD7_1000BASE_T_POWER_SAVE_PER_CABLE_LENGTH);
|
||||
+ control_dac &= ~QCA807X_CONTROL_DAC_MASK;
|
||||
+ if (!priv->dac_full_amplitude)
|
||||
+ control_dac |= QCA807X_CONTROL_DAC_DSP_AMPLITUDE;
|
||||
+ if (!priv->dac_full_amplitude)
|
||||
+ control_dac |= QCA807X_CONTROL_DAC_DSP_BIAS_CURRENT;
|
||||
+ if (!priv->dac_disable_bias_current_tweak)
|
||||
+ control_dac |= QCA807X_CONTROL_DAC_BIAS_CURRENT_TWEAK;
|
||||
+ return phy_write_mmd(phydev, MDIO_MMD_AN,
|
||||
+ QCA807X_MMD7_1000BASE_T_POWER_SAVE_PER_CABLE_LENGTH,
|
||||
+ control_dac);
|
||||
+}
|
||||
+
|
||||
+static struct phy_driver qca807x_drivers[] = {
|
||||
+ {
|
||||
+ PHY_ID_MATCH_EXACT(PHY_ID_QCA8072),
|
||||
+ .name = "Qualcomm QCA8072",
|
||||
+ .flags = PHY_POLL_CABLE_TEST,
|
||||
+ /* PHY_GBIT_FEATURES */
|
||||
+ .probe = qca807x_probe,
|
||||
+ .config_init = qca807x_config_init,
|
||||
+ .read_status = qca807x_read_status,
|
||||
+ .config_intr = at803x_config_intr,
|
||||
+ .handle_interrupt = at803x_handle_interrupt,
|
||||
+ .soft_reset = genphy_soft_reset,
|
||||
+ .get_tunable = at803x_get_tunable,
|
||||
+ .set_tunable = at803x_set_tunable,
|
||||
+ .resume = genphy_resume,
|
||||
+ .suspend = genphy_suspend,
|
||||
+ .cable_test_start = qca807x_cable_test_start,
|
||||
+ .cable_test_get_status = qca808x_cable_test_get_status,
|
||||
+ },
|
||||
+ {
|
||||
+ PHY_ID_MATCH_EXACT(PHY_ID_QCA8075),
|
||||
+ .name = "Qualcomm QCA8075",
|
||||
+ .flags = PHY_POLL_CABLE_TEST,
|
||||
+ /* PHY_GBIT_FEATURES */
|
||||
+ .probe = qca807x_probe,
|
||||
+ .config_init = qca807x_config_init,
|
||||
+ .read_status = qca807x_read_status,
|
||||
+ .config_intr = at803x_config_intr,
|
||||
+ .handle_interrupt = at803x_handle_interrupt,
|
||||
+ .soft_reset = genphy_soft_reset,
|
||||
+ .get_tunable = at803x_get_tunable,
|
||||
+ .set_tunable = at803x_set_tunable,
|
||||
+ .resume = genphy_resume,
|
||||
+ .suspend = genphy_suspend,
|
||||
+ .cable_test_start = qca807x_cable_test_start,
|
||||
+ .cable_test_get_status = qca808x_cable_test_get_status,
|
||||
+ },
|
||||
+};
|
||||
+module_phy_driver(qca807x_drivers);
|
||||
+
|
||||
+static struct mdio_device_id __maybe_unused qca807x_tbl[] = {
|
||||
+ { PHY_ID_MATCH_EXACT(PHY_ID_QCA8072) },
|
||||
+ { PHY_ID_MATCH_EXACT(PHY_ID_QCA8075) },
|
||||
+ { }
|
||||
+};
|
||||
+
|
||||
+MODULE_AUTHOR("Robert Marko <robert.marko@sartura.hr>");
|
||||
+MODULE_AUTHOR("Christian Marangi <ansuelsmth@gmail.com>");
|
||||
+MODULE_DESCRIPTION("Qualcomm QCA807x PHY driver");
|
||||
+MODULE_DEVICE_TABLE(mdio, qca807x_tbl);
|
||||
+MODULE_LICENSE("GPL");
|
@ -0,0 +1,179 @@
|
||||
From ee9d9807bee0e6af8ca2a4db6f0d1dc0e5b41f44 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 6 Feb 2024 18:31:11 +0100
|
||||
Subject: [PATCH 08/10] net: phy: qcom: move common qca808x LED define to
|
||||
shared header
|
||||
|
||||
The LED implementation of qca808x and qca807x is the same but qca807x
|
||||
supports also Fiber port and have different hw control bits for Fiber
|
||||
port.
|
||||
|
||||
In preparation for qca807x introduction, move all the common define to
|
||||
shared header.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/qcom/qca808x.c | 65 ----------------------------------
|
||||
drivers/net/phy/qcom/qcom.h | 65 ++++++++++++++++++++++++++++++++++
|
||||
2 files changed, 65 insertions(+), 65 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/qcom/qca808x.c
|
||||
+++ b/drivers/net/phy/qcom/qca808x.c
|
||||
@@ -62,29 +62,6 @@
|
||||
#define QCA808X_DBG_AN_TEST 0xb
|
||||
#define QCA808X_HIBERNATION_EN BIT(15)
|
||||
|
||||
-#define QCA808X_MMD7_LED_GLOBAL 0x8073
|
||||
-#define QCA808X_LED_BLINK_1 GENMASK(11, 6)
|
||||
-#define QCA808X_LED_BLINK_2 GENMASK(5, 0)
|
||||
-/* Values are the same for both BLINK_1 and BLINK_2 */
|
||||
-#define QCA808X_LED_BLINK_FREQ_MASK GENMASK(5, 3)
|
||||
-#define QCA808X_LED_BLINK_FREQ_2HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x0)
|
||||
-#define QCA808X_LED_BLINK_FREQ_4HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x1)
|
||||
-#define QCA808X_LED_BLINK_FREQ_8HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x2)
|
||||
-#define QCA808X_LED_BLINK_FREQ_16HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x3)
|
||||
-#define QCA808X_LED_BLINK_FREQ_32HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x4)
|
||||
-#define QCA808X_LED_BLINK_FREQ_64HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x5)
|
||||
-#define QCA808X_LED_BLINK_FREQ_128HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x6)
|
||||
-#define QCA808X_LED_BLINK_FREQ_256HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x7)
|
||||
-#define QCA808X_LED_BLINK_DUTY_MASK GENMASK(2, 0)
|
||||
-#define QCA808X_LED_BLINK_DUTY_50_50 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x0)
|
||||
-#define QCA808X_LED_BLINK_DUTY_75_25 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x1)
|
||||
-#define QCA808X_LED_BLINK_DUTY_25_75 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x2)
|
||||
-#define QCA808X_LED_BLINK_DUTY_33_67 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x3)
|
||||
-#define QCA808X_LED_BLINK_DUTY_67_33 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x4)
|
||||
-#define QCA808X_LED_BLINK_DUTY_17_83 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x5)
|
||||
-#define QCA808X_LED_BLINK_DUTY_83_17 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x6)
|
||||
-#define QCA808X_LED_BLINK_DUTY_8_92 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x7)
|
||||
-
|
||||
#define QCA808X_MMD7_LED2_CTRL 0x8074
|
||||
#define QCA808X_MMD7_LED2_FORCE_CTRL 0x8075
|
||||
#define QCA808X_MMD7_LED1_CTRL 0x8076
|
||||
@@ -92,51 +69,9 @@
|
||||
#define QCA808X_MMD7_LED0_CTRL 0x8078
|
||||
#define QCA808X_MMD7_LED_CTRL(x) (0x8078 - ((x) * 2))
|
||||
|
||||
-/* LED hw control pattern is the same for every LED */
|
||||
-#define QCA808X_LED_PATTERN_MASK GENMASK(15, 0)
|
||||
-#define QCA808X_LED_SPEED2500_ON BIT(15)
|
||||
-#define QCA808X_LED_SPEED2500_BLINK BIT(14)
|
||||
-/* Follow blink trigger even if duplex or speed condition doesn't match */
|
||||
-#define QCA808X_LED_BLINK_CHECK_BYPASS BIT(13)
|
||||
-#define QCA808X_LED_FULL_DUPLEX_ON BIT(12)
|
||||
-#define QCA808X_LED_HALF_DUPLEX_ON BIT(11)
|
||||
-#define QCA808X_LED_TX_BLINK BIT(10)
|
||||
-#define QCA808X_LED_RX_BLINK BIT(9)
|
||||
-#define QCA808X_LED_TX_ON_10MS BIT(8)
|
||||
-#define QCA808X_LED_RX_ON_10MS BIT(7)
|
||||
-#define QCA808X_LED_SPEED1000_ON BIT(6)
|
||||
-#define QCA808X_LED_SPEED100_ON BIT(5)
|
||||
-#define QCA808X_LED_SPEED10_ON BIT(4)
|
||||
-#define QCA808X_LED_COLLISION_BLINK BIT(3)
|
||||
-#define QCA808X_LED_SPEED1000_BLINK BIT(2)
|
||||
-#define QCA808X_LED_SPEED100_BLINK BIT(1)
|
||||
-#define QCA808X_LED_SPEED10_BLINK BIT(0)
|
||||
-
|
||||
#define QCA808X_MMD7_LED0_FORCE_CTRL 0x8079
|
||||
#define QCA808X_MMD7_LED_FORCE_CTRL(x) (0x8079 - ((x) * 2))
|
||||
|
||||
-/* LED force ctrl is the same for every LED
|
||||
- * No documentation exist for this, not even internal one
|
||||
- * with NDA as QCOM gives only info about configuring
|
||||
- * hw control pattern rules and doesn't indicate any way
|
||||
- * to force the LED to specific mode.
|
||||
- * These define comes from reverse and testing and maybe
|
||||
- * lack of some info or some info are not entirely correct.
|
||||
- * For the basic LED control and hw control these finding
|
||||
- * are enough to support LED control in all the required APIs.
|
||||
- *
|
||||
- * On doing some comparison with implementation with qca807x,
|
||||
- * it was found that it's 1:1 equal to it and confirms all the
|
||||
- * reverse done. It was also found further specification with the
|
||||
- * force mode and the blink modes.
|
||||
- */
|
||||
-#define QCA808X_LED_FORCE_EN BIT(15)
|
||||
-#define QCA808X_LED_FORCE_MODE_MASK GENMASK(14, 13)
|
||||
-#define QCA808X_LED_FORCE_BLINK_1 FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x3)
|
||||
-#define QCA808X_LED_FORCE_BLINK_2 FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x2)
|
||||
-#define QCA808X_LED_FORCE_ON FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x1)
|
||||
-#define QCA808X_LED_FORCE_OFF FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x0)
|
||||
-
|
||||
#define QCA808X_MMD7_LED_POLARITY_CTRL 0x901a
|
||||
/* QSDK sets by default 0x46 to this reg that sets BIT 6 for
|
||||
* LED to active high. It's not clear what BIT 3 and BIT 4 does.
|
||||
--- a/drivers/net/phy/qcom/qcom.h
|
||||
+++ b/drivers/net/phy/qcom/qcom.h
|
||||
@@ -103,6 +103,71 @@
|
||||
/* Added for reference of existence but should be handled by wait_for_completion already */
|
||||
#define QCA808X_CDT_STATUS_STAT_BUSY (BIT(1) | BIT(3))
|
||||
|
||||
+#define QCA808X_MMD7_LED_GLOBAL 0x8073
|
||||
+#define QCA808X_LED_BLINK_1 GENMASK(11, 6)
|
||||
+#define QCA808X_LED_BLINK_2 GENMASK(5, 0)
|
||||
+/* Values are the same for both BLINK_1 and BLINK_2 */
|
||||
+#define QCA808X_LED_BLINK_FREQ_MASK GENMASK(5, 3)
|
||||
+#define QCA808X_LED_BLINK_FREQ_2HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x0)
|
||||
+#define QCA808X_LED_BLINK_FREQ_4HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x1)
|
||||
+#define QCA808X_LED_BLINK_FREQ_8HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x2)
|
||||
+#define QCA808X_LED_BLINK_FREQ_16HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x3)
|
||||
+#define QCA808X_LED_BLINK_FREQ_32HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x4)
|
||||
+#define QCA808X_LED_BLINK_FREQ_64HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x5)
|
||||
+#define QCA808X_LED_BLINK_FREQ_128HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x6)
|
||||
+#define QCA808X_LED_BLINK_FREQ_256HZ FIELD_PREP(QCA808X_LED_BLINK_FREQ_MASK, 0x7)
|
||||
+#define QCA808X_LED_BLINK_DUTY_MASK GENMASK(2, 0)
|
||||
+#define QCA808X_LED_BLINK_DUTY_50_50 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x0)
|
||||
+#define QCA808X_LED_BLINK_DUTY_75_25 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x1)
|
||||
+#define QCA808X_LED_BLINK_DUTY_25_75 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x2)
|
||||
+#define QCA808X_LED_BLINK_DUTY_33_67 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x3)
|
||||
+#define QCA808X_LED_BLINK_DUTY_67_33 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x4)
|
||||
+#define QCA808X_LED_BLINK_DUTY_17_83 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x5)
|
||||
+#define QCA808X_LED_BLINK_DUTY_83_17 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x6)
|
||||
+#define QCA808X_LED_BLINK_DUTY_8_92 FIELD_PREP(QCA808X_LED_BLINK_DUTY_MASK, 0x7)
|
||||
+
|
||||
+/* LED hw control pattern is the same for every LED */
|
||||
+#define QCA808X_LED_PATTERN_MASK GENMASK(15, 0)
|
||||
+#define QCA808X_LED_SPEED2500_ON BIT(15)
|
||||
+#define QCA808X_LED_SPEED2500_BLINK BIT(14)
|
||||
+/* Follow blink trigger even if duplex or speed condition doesn't match */
|
||||
+#define QCA808X_LED_BLINK_CHECK_BYPASS BIT(13)
|
||||
+#define QCA808X_LED_FULL_DUPLEX_ON BIT(12)
|
||||
+#define QCA808X_LED_HALF_DUPLEX_ON BIT(11)
|
||||
+#define QCA808X_LED_TX_BLINK BIT(10)
|
||||
+#define QCA808X_LED_RX_BLINK BIT(9)
|
||||
+#define QCA808X_LED_TX_ON_10MS BIT(8)
|
||||
+#define QCA808X_LED_RX_ON_10MS BIT(7)
|
||||
+#define QCA808X_LED_SPEED1000_ON BIT(6)
|
||||
+#define QCA808X_LED_SPEED100_ON BIT(5)
|
||||
+#define QCA808X_LED_SPEED10_ON BIT(4)
|
||||
+#define QCA808X_LED_COLLISION_BLINK BIT(3)
|
||||
+#define QCA808X_LED_SPEED1000_BLINK BIT(2)
|
||||
+#define QCA808X_LED_SPEED100_BLINK BIT(1)
|
||||
+#define QCA808X_LED_SPEED10_BLINK BIT(0)
|
||||
+
|
||||
+/* LED force ctrl is the same for every LED
|
||||
+ * No documentation exist for this, not even internal one
|
||||
+ * with NDA as QCOM gives only info about configuring
|
||||
+ * hw control pattern rules and doesn't indicate any way
|
||||
+ * to force the LED to specific mode.
|
||||
+ * These define comes from reverse and testing and maybe
|
||||
+ * lack of some info or some info are not entirely correct.
|
||||
+ * For the basic LED control and hw control these finding
|
||||
+ * are enough to support LED control in all the required APIs.
|
||||
+ *
|
||||
+ * On doing some comparison with implementation with qca807x,
|
||||
+ * it was found that it's 1:1 equal to it and confirms all the
|
||||
+ * reverse done. It was also found further specification with the
|
||||
+ * force mode and the blink modes.
|
||||
+ */
|
||||
+#define QCA808X_LED_FORCE_EN BIT(15)
|
||||
+#define QCA808X_LED_FORCE_MODE_MASK GENMASK(14, 13)
|
||||
+#define QCA808X_LED_FORCE_BLINK_1 FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x3)
|
||||
+#define QCA808X_LED_FORCE_BLINK_2 FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x2)
|
||||
+#define QCA808X_LED_FORCE_ON FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x1)
|
||||
+#define QCA808X_LED_FORCE_OFF FIELD_PREP(QCA808X_LED_FORCE_MODE_MASK, 0x0)
|
||||
+
|
||||
#define AT803X_LOC_MAC_ADDR_0_15_OFFSET 0x804C
|
||||
#define AT803X_LOC_MAC_ADDR_16_31_OFFSET 0x804B
|
||||
#define AT803X_LOC_MAC_ADDR_32_47_OFFSET 0x804A
|
@ -0,0 +1,172 @@
|
||||
From 47b930d0dd437af927145dba50a2e2ea1ba97c67 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 6 Feb 2024 18:31:12 +0100
|
||||
Subject: [PATCH 09/10] net: phy: qcom: generalize some qca808x LED functions
|
||||
|
||||
Generalize some qca808x LED functions in preparation for qca807x LED
|
||||
support.
|
||||
|
||||
The LED implementation of qca808x and qca807x is the same but qca807x
|
||||
supports also Fiber port and have different hw control bits for Fiber
|
||||
port. To limit code duplication introduce micro functions that takes reg
|
||||
instead of LED index to tweak all the supported LED modes.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/qcom/qca808x.c | 38 +++-----------------
|
||||
drivers/net/phy/qcom/qcom-phy-lib.c | 54 +++++++++++++++++++++++++++++
|
||||
drivers/net/phy/qcom/qcom.h | 7 ++++
|
||||
3 files changed, 65 insertions(+), 34 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/qcom/qca808x.c
|
||||
+++ b/drivers/net/phy/qcom/qca808x.c
|
||||
@@ -437,9 +437,7 @@ static int qca808x_led_hw_control_enable
|
||||
return -EINVAL;
|
||||
|
||||
reg = QCA808X_MMD7_LED_FORCE_CTRL(index);
|
||||
-
|
||||
- return phy_clear_bits_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
- QCA808X_LED_FORCE_EN);
|
||||
+ return qca808x_led_reg_hw_control_enable(phydev, reg);
|
||||
}
|
||||
|
||||
static int qca808x_led_hw_is_supported(struct phy_device *phydev, u8 index,
|
||||
@@ -480,16 +478,12 @@ static int qca808x_led_hw_control_set(st
|
||||
static bool qca808x_led_hw_control_status(struct phy_device *phydev, u8 index)
|
||||
{
|
||||
u16 reg;
|
||||
- int val;
|
||||
|
||||
if (index > 2)
|
||||
return false;
|
||||
|
||||
reg = QCA808X_MMD7_LED_FORCE_CTRL(index);
|
||||
-
|
||||
- val = phy_read_mmd(phydev, MDIO_MMD_AN, reg);
|
||||
-
|
||||
- return !(val & QCA808X_LED_FORCE_EN);
|
||||
+ return qca808x_led_reg_hw_control_status(phydev, reg);
|
||||
}
|
||||
|
||||
static int qca808x_led_hw_control_get(struct phy_device *phydev, u8 index,
|
||||
@@ -557,44 +551,20 @@ static int qca808x_led_brightness_set(st
|
||||
}
|
||||
|
||||
reg = QCA808X_MMD7_LED_FORCE_CTRL(index);
|
||||
-
|
||||
- return phy_modify_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
- QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_MODE_MASK,
|
||||
- QCA808X_LED_FORCE_EN | (value ? QCA808X_LED_FORCE_ON :
|
||||
- QCA808X_LED_FORCE_OFF));
|
||||
+ return qca808x_led_reg_brightness_set(phydev, reg, value);
|
||||
}
|
||||
|
||||
static int qca808x_led_blink_set(struct phy_device *phydev, u8 index,
|
||||
unsigned long *delay_on,
|
||||
unsigned long *delay_off)
|
||||
{
|
||||
- int ret;
|
||||
u16 reg;
|
||||
|
||||
if (index > 2)
|
||||
return -EINVAL;
|
||||
|
||||
reg = QCA808X_MMD7_LED_FORCE_CTRL(index);
|
||||
-
|
||||
- /* Set blink to 50% off, 50% on at 4Hz by default */
|
||||
- ret = phy_modify_mmd(phydev, MDIO_MMD_AN, QCA808X_MMD7_LED_GLOBAL,
|
||||
- QCA808X_LED_BLINK_FREQ_MASK | QCA808X_LED_BLINK_DUTY_MASK,
|
||||
- QCA808X_LED_BLINK_FREQ_4HZ | QCA808X_LED_BLINK_DUTY_50_50);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- /* We use BLINK_1 for normal blinking */
|
||||
- ret = phy_modify_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
- QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_MODE_MASK,
|
||||
- QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_BLINK_1);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
- /* We set blink to 4Hz, aka 250ms */
|
||||
- *delay_on = 250 / 2;
|
||||
- *delay_off = 250 / 2;
|
||||
-
|
||||
- return 0;
|
||||
+ return qca808x_led_reg_blink_set(phydev, reg, delay_on, delay_off);
|
||||
}
|
||||
|
||||
static int qca808x_led_polarity_set(struct phy_device *phydev, int index,
|
||||
--- a/drivers/net/phy/qcom/qcom-phy-lib.c
|
||||
+++ b/drivers/net/phy/qcom/qcom-phy-lib.c
|
||||
@@ -620,3 +620,57 @@ int qca808x_cable_test_get_status(struct
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(qca808x_cable_test_get_status);
|
||||
+
|
||||
+int qca808x_led_reg_hw_control_enable(struct phy_device *phydev, u16 reg)
|
||||
+{
|
||||
+ return phy_clear_bits_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
+ QCA808X_LED_FORCE_EN);
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(qca808x_led_reg_hw_control_enable);
|
||||
+
|
||||
+bool qca808x_led_reg_hw_control_status(struct phy_device *phydev, u16 reg)
|
||||
+{
|
||||
+ int val;
|
||||
+
|
||||
+ val = phy_read_mmd(phydev, MDIO_MMD_AN, reg);
|
||||
+ return !(val & QCA808X_LED_FORCE_EN);
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(qca808x_led_reg_hw_control_status);
|
||||
+
|
||||
+int qca808x_led_reg_brightness_set(struct phy_device *phydev,
|
||||
+ u16 reg, enum led_brightness value)
|
||||
+{
|
||||
+ return phy_modify_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
+ QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_MODE_MASK,
|
||||
+ QCA808X_LED_FORCE_EN | (value ? QCA808X_LED_FORCE_ON :
|
||||
+ QCA808X_LED_FORCE_OFF));
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(qca808x_led_reg_brightness_set);
|
||||
+
|
||||
+int qca808x_led_reg_blink_set(struct phy_device *phydev, u16 reg,
|
||||
+ unsigned long *delay_on,
|
||||
+ unsigned long *delay_off)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ /* Set blink to 50% off, 50% on at 4Hz by default */
|
||||
+ ret = phy_modify_mmd(phydev, MDIO_MMD_AN, QCA808X_MMD7_LED_GLOBAL,
|
||||
+ QCA808X_LED_BLINK_FREQ_MASK | QCA808X_LED_BLINK_DUTY_MASK,
|
||||
+ QCA808X_LED_BLINK_FREQ_4HZ | QCA808X_LED_BLINK_DUTY_50_50);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ /* We use BLINK_1 for normal blinking */
|
||||
+ ret = phy_modify_mmd(phydev, MDIO_MMD_AN, reg,
|
||||
+ QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_MODE_MASK,
|
||||
+ QCA808X_LED_FORCE_EN | QCA808X_LED_FORCE_BLINK_1);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ /* We set blink to 4Hz, aka 250ms */
|
||||
+ *delay_on = 250 / 2;
|
||||
+ *delay_off = 250 / 2;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(qca808x_led_reg_blink_set);
|
||||
--- a/drivers/net/phy/qcom/qcom.h
|
||||
+++ b/drivers/net/phy/qcom/qcom.h
|
||||
@@ -234,3 +234,10 @@ int at803x_cdt_start(struct phy_device *
|
||||
int at803x_cdt_wait_for_completion(struct phy_device *phydev,
|
||||
u32 cdt_en);
|
||||
int qca808x_cable_test_get_status(struct phy_device *phydev, bool *finished);
|
||||
+int qca808x_led_reg_hw_control_enable(struct phy_device *phydev, u16 reg);
|
||||
+bool qca808x_led_reg_hw_control_status(struct phy_device *phydev, u16 reg);
|
||||
+int qca808x_led_reg_brightness_set(struct phy_device *phydev,
|
||||
+ u16 reg, enum led_brightness value);
|
||||
+int qca808x_led_reg_blink_set(struct phy_device *phydev, u16 reg,
|
||||
+ unsigned long *delay_on,
|
||||
+ unsigned long *delay_off);
|
@ -0,0 +1,326 @@
|
||||
From f508a226b517a6a8afd78a317de46bc83e3e3d51 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 6 Feb 2024 18:31:13 +0100
|
||||
Subject: [PATCH 10/10] net: phy: qca807x: add support for configurable LED
|
||||
|
||||
QCA8072/5 have up to 2 LEDs attached for PHY.
|
||||
|
||||
LEDs can be configured to be ON/hw blink or be set to HW control.
|
||||
|
||||
Hw blink mode is set to blink at 4Hz or 250ms.
|
||||
|
||||
PHY can support both copper (TP) or fiber (FIBRE) kind and supports
|
||||
different HW control modes based on the port type.
|
||||
|
||||
HW control modes supported for netdev trigger for copper ports are:
|
||||
- LINK_10
|
||||
- LINK_100
|
||||
- LINK_1000
|
||||
- TX
|
||||
- RX
|
||||
- FULL_DUPLEX
|
||||
- HALF_DUPLEX
|
||||
|
||||
HW control modes supported for netdev trigger for fiber ports are:
|
||||
- LINK_100
|
||||
- LINK_1000
|
||||
- TX
|
||||
- RX
|
||||
- FULL_DUPLEX
|
||||
- HALF_DUPLEX
|
||||
|
||||
LED support conflicts with GPIO controller feature and must be disabled
|
||||
if gpio-controller is used for the PHY.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/qcom/qca807x.c | 256 ++++++++++++++++++++++++++++++++-
|
||||
1 file changed, 254 insertions(+), 2 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/qcom/qca807x.c
|
||||
+++ b/drivers/net/phy/qcom/qca807x.c
|
||||
@@ -57,8 +57,18 @@
|
||||
#define QCA807X_MMD7_LED_CTRL(x) (0x8074 + ((x) * 2))
|
||||
#define QCA807X_MMD7_LED_FORCE_CTRL(x) (0x8075 + ((x) * 2))
|
||||
|
||||
-#define QCA807X_GPIO_FORCE_EN BIT(15)
|
||||
-#define QCA807X_GPIO_FORCE_MODE_MASK GENMASK(14, 13)
|
||||
+/* LED hw control pattern for fiber port */
|
||||
+#define QCA807X_LED_FIBER_PATTERN_MASK GENMASK(11, 1)
|
||||
+#define QCA807X_LED_FIBER_TXACT_BLK_EN BIT(10)
|
||||
+#define QCA807X_LED_FIBER_RXACT_BLK_EN BIT(9)
|
||||
+#define QCA807X_LED_FIBER_FDX_ON_EN BIT(6)
|
||||
+#define QCA807X_LED_FIBER_HDX_ON_EN BIT(5)
|
||||
+#define QCA807X_LED_FIBER_1000BX_ON_EN BIT(2)
|
||||
+#define QCA807X_LED_FIBER_100FX_ON_EN BIT(1)
|
||||
+
|
||||
+/* Some device repurpose the LED as GPIO out */
|
||||
+#define QCA807X_GPIO_FORCE_EN QCA808X_LED_FORCE_EN
|
||||
+#define QCA807X_GPIO_FORCE_MODE_MASK QCA808X_LED_FORCE_MODE_MASK
|
||||
|
||||
#define QCA807X_FUNCTION_CONTROL 0x10
|
||||
#define QCA807X_FC_MDI_CROSSOVER_MODE_MASK GENMASK(6, 5)
|
||||
@@ -121,6 +131,233 @@ static int qca807x_cable_test_start(stru
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int qca807x_led_parse_netdev(struct phy_device *phydev, unsigned long rules,
|
||||
+ u16 *offload_trigger)
|
||||
+{
|
||||
+ /* Parsing specific to netdev trigger */
|
||||
+ switch (phydev->port) {
|
||||
+ case PORT_TP:
|
||||
+ if (test_bit(TRIGGER_NETDEV_TX, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_TX_BLINK;
|
||||
+ if (test_bit(TRIGGER_NETDEV_RX, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_RX_BLINK;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_10, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_SPEED10_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_100, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_SPEED100_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_1000, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_SPEED1000_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_HALF_DUPLEX, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_HALF_DUPLEX_ON;
|
||||
+ if (test_bit(TRIGGER_NETDEV_FULL_DUPLEX, &rules))
|
||||
+ *offload_trigger |= QCA808X_LED_FULL_DUPLEX_ON;
|
||||
+ break;
|
||||
+ case PORT_FIBRE:
|
||||
+ if (test_bit(TRIGGER_NETDEV_TX, &rules))
|
||||
+ *offload_trigger |= QCA807X_LED_FIBER_TXACT_BLK_EN;
|
||||
+ if (test_bit(TRIGGER_NETDEV_RX, &rules))
|
||||
+ *offload_trigger |= QCA807X_LED_FIBER_RXACT_BLK_EN;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_100, &rules))
|
||||
+ *offload_trigger |= QCA807X_LED_FIBER_100FX_ON_EN;
|
||||
+ if (test_bit(TRIGGER_NETDEV_LINK_1000, &rules))
|
||||
+ *offload_trigger |= QCA807X_LED_FIBER_1000BX_ON_EN;
|
||||
+ if (test_bit(TRIGGER_NETDEV_HALF_DUPLEX, &rules))
|
||||
+ *offload_trigger |= QCA807X_LED_FIBER_HDX_ON_EN;
|
||||
+ if (test_bit(TRIGGER_NETDEV_FULL_DUPLEX, &rules))
|
||||
+ *offload_trigger |= QCA807X_LED_FIBER_FDX_ON_EN;
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EOPNOTSUPP;
|
||||
+ }
|
||||
+
|
||||
+ if (rules && !*offload_trigger)
|
||||
+ return -EOPNOTSUPP;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca807x_led_hw_control_enable(struct phy_device *phydev, u8 index)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+
|
||||
+ if (index > 1)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(index);
|
||||
+ return qca808x_led_reg_hw_control_enable(phydev, reg);
|
||||
+}
|
||||
+
|
||||
+static int qca807x_led_hw_is_supported(struct phy_device *phydev, u8 index,
|
||||
+ unsigned long rules)
|
||||
+{
|
||||
+ u16 offload_trigger = 0;
|
||||
+
|
||||
+ if (index > 1)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ return qca807x_led_parse_netdev(phydev, rules, &offload_trigger);
|
||||
+}
|
||||
+
|
||||
+static int qca807x_led_hw_control_set(struct phy_device *phydev, u8 index,
|
||||
+ unsigned long rules)
|
||||
+{
|
||||
+ u16 reg, mask, offload_trigger = 0;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (index > 1)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ ret = qca807x_led_parse_netdev(phydev, rules, &offload_trigger);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = qca807x_led_hw_control_enable(phydev, index);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ switch (phydev->port) {
|
||||
+ case PORT_TP:
|
||||
+ reg = QCA807X_MMD7_LED_CTRL(index);
|
||||
+ mask = QCA808X_LED_PATTERN_MASK;
|
||||
+ break;
|
||||
+ case PORT_FIBRE:
|
||||
+ /* HW control pattern bits are in LED FORCE reg */
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(index);
|
||||
+ mask = QCA807X_LED_FIBER_PATTERN_MASK;
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ return phy_modify_mmd(phydev, MDIO_MMD_AN, reg, mask,
|
||||
+ offload_trigger);
|
||||
+}
|
||||
+
|
||||
+static bool qca807x_led_hw_control_status(struct phy_device *phydev, u8 index)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+
|
||||
+ if (index > 1)
|
||||
+ return false;
|
||||
+
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(index);
|
||||
+ return qca808x_led_reg_hw_control_status(phydev, reg);
|
||||
+}
|
||||
+
|
||||
+static int qca807x_led_hw_control_get(struct phy_device *phydev, u8 index,
|
||||
+ unsigned long *rules)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+ int val;
|
||||
+
|
||||
+ if (index > 1)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ /* Check if we have hw control enabled */
|
||||
+ if (qca807x_led_hw_control_status(phydev, index))
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ /* Parsing specific to netdev trigger */
|
||||
+ switch (phydev->port) {
|
||||
+ case PORT_TP:
|
||||
+ reg = QCA807X_MMD7_LED_CTRL(index);
|
||||
+ val = phy_read_mmd(phydev, MDIO_MMD_AN, reg);
|
||||
+ if (val & QCA808X_LED_TX_BLINK)
|
||||
+ set_bit(TRIGGER_NETDEV_TX, rules);
|
||||
+ if (val & QCA808X_LED_RX_BLINK)
|
||||
+ set_bit(TRIGGER_NETDEV_RX, rules);
|
||||
+ if (val & QCA808X_LED_SPEED10_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_10, rules);
|
||||
+ if (val & QCA808X_LED_SPEED100_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_100, rules);
|
||||
+ if (val & QCA808X_LED_SPEED1000_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_1000, rules);
|
||||
+ if (val & QCA808X_LED_HALF_DUPLEX_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_HALF_DUPLEX, rules);
|
||||
+ if (val & QCA808X_LED_FULL_DUPLEX_ON)
|
||||
+ set_bit(TRIGGER_NETDEV_FULL_DUPLEX, rules);
|
||||
+ break;
|
||||
+ case PORT_FIBRE:
|
||||
+ /* HW control pattern bits are in LED FORCE reg */
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(index);
|
||||
+ val = phy_read_mmd(phydev, MDIO_MMD_AN, reg);
|
||||
+ if (val & QCA807X_LED_FIBER_TXACT_BLK_EN)
|
||||
+ set_bit(TRIGGER_NETDEV_TX, rules);
|
||||
+ if (val & QCA807X_LED_FIBER_RXACT_BLK_EN)
|
||||
+ set_bit(TRIGGER_NETDEV_RX, rules);
|
||||
+ if (val & QCA807X_LED_FIBER_100FX_ON_EN)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_100, rules);
|
||||
+ if (val & QCA807X_LED_FIBER_1000BX_ON_EN)
|
||||
+ set_bit(TRIGGER_NETDEV_LINK_1000, rules);
|
||||
+ if (val & QCA807X_LED_FIBER_HDX_ON_EN)
|
||||
+ set_bit(TRIGGER_NETDEV_HALF_DUPLEX, rules);
|
||||
+ if (val & QCA807X_LED_FIBER_FDX_ON_EN)
|
||||
+ set_bit(TRIGGER_NETDEV_FULL_DUPLEX, rules);
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int qca807x_led_hw_control_reset(struct phy_device *phydev, u8 index)
|
||||
+{
|
||||
+ u16 reg, mask;
|
||||
+
|
||||
+ if (index > 1)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ switch (phydev->port) {
|
||||
+ case PORT_TP:
|
||||
+ reg = QCA807X_MMD7_LED_CTRL(index);
|
||||
+ mask = QCA808X_LED_PATTERN_MASK;
|
||||
+ break;
|
||||
+ case PORT_FIBRE:
|
||||
+ /* HW control pattern bits are in LED FORCE reg */
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(index);
|
||||
+ mask = QCA807X_LED_FIBER_PATTERN_MASK;
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ return phy_clear_bits_mmd(phydev, MDIO_MMD_AN, reg, mask);
|
||||
+}
|
||||
+
|
||||
+static int qca807x_led_brightness_set(struct phy_device *phydev,
|
||||
+ u8 index, enum led_brightness value)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (index > 1)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ /* If we are setting off the LED reset any hw control rule */
|
||||
+ if (!value) {
|
||||
+ ret = qca807x_led_hw_control_reset(phydev, index);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(index);
|
||||
+ return qca808x_led_reg_brightness_set(phydev, reg, value);
|
||||
+}
|
||||
+
|
||||
+static int qca807x_led_blink_set(struct phy_device *phydev, u8 index,
|
||||
+ unsigned long *delay_on,
|
||||
+ unsigned long *delay_off)
|
||||
+{
|
||||
+ u16 reg;
|
||||
+
|
||||
+ if (index > 1)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ reg = QCA807X_MMD7_LED_FORCE_CTRL(index);
|
||||
+ return qca808x_led_reg_blink_set(phydev, reg, delay_on, delay_off);
|
||||
+}
|
||||
+
|
||||
#ifdef CONFIG_GPIOLIB
|
||||
static int qca807x_gpio_get_direction(struct gpio_chip *gc, unsigned int offset)
|
||||
{
|
||||
@@ -496,6 +733,16 @@ static int qca807x_probe(struct phy_devi
|
||||
"qcom,dac-disable-bias-current-tweak");
|
||||
|
||||
if (IS_ENABLED(CONFIG_GPIOLIB)) {
|
||||
+ /* Make sure we don't have mixed leds node and gpio-controller
|
||||
+ * to prevent registering leds and having gpio-controller usage
|
||||
+ * conflicting with them.
|
||||
+ */
|
||||
+ if (of_find_property(node, "leds", NULL) &&
|
||||
+ of_find_property(node, "gpio-controller", NULL)) {
|
||||
+ phydev_err(phydev, "Invalid property detected. LEDs and gpio-controller are mutually exclusive.");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
/* Do not register a GPIO controller unless flagged for it */
|
||||
if (of_property_read_bool(node, "gpio-controller")) {
|
||||
ret = qca807x_gpio(phydev);
|
||||
@@ -580,6 +827,11 @@ static struct phy_driver qca807x_drivers
|
||||
.suspend = genphy_suspend,
|
||||
.cable_test_start = qca807x_cable_test_start,
|
||||
.cable_test_get_status = qca808x_cable_test_get_status,
|
||||
+ .led_brightness_set = qca807x_led_brightness_set,
|
||||
+ .led_blink_set = qca807x_led_blink_set,
|
||||
+ .led_hw_is_supported = qca807x_led_hw_is_supported,
|
||||
+ .led_hw_control_set = qca807x_led_hw_control_set,
|
||||
+ .led_hw_control_get = qca807x_led_hw_control_get,
|
||||
},
|
||||
};
|
||||
module_phy_driver(qca807x_drivers);
|
@ -0,0 +1,51 @@
|
||||
From 3be0d950b62852a693182cb678948f481de02825 Mon Sep 17 00:00:00 2001
|
||||
From: Robert Marko <robimarko@gmail.com>
|
||||
Date: Mon, 12 Feb 2024 12:49:34 +0100
|
||||
Subject: [PATCH] net: phy: qca807x: move interface mode check to
|
||||
.config_init_once
|
||||
|
||||
Currently, we are checking whether the PHY package mode matches the
|
||||
individual PHY interface modes at PHY package probe time, but at that time
|
||||
we only know the PHY package mode and not the individual PHY interface
|
||||
modes as of_get_phy_mode() that populates it will only get called once the
|
||||
netdev to which PHY-s are attached to is being probed and thus this check
|
||||
will always fail and return -EINVAL.
|
||||
|
||||
So, lets move this check to .config_init_once as at that point individual
|
||||
PHY interface modes should be populated.
|
||||
|
||||
Fixes: d1cb613efbd3 ("net: phy: qcom: add support for QCA807x PHY Family")
|
||||
Signed-off-by: Robert Marko <robimarko@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Link: https://lore.kernel.org/r/20240212115043.1725918-1-robimarko@gmail.com
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
drivers/net/phy/qcom/qca807x.c | 10 +++++-----
|
||||
1 file changed, 5 insertions(+), 5 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/qcom/qca807x.c
|
||||
+++ b/drivers/net/phy/qcom/qca807x.c
|
||||
@@ -562,6 +562,11 @@ static int qca807x_phy_package_config_in
|
||||
struct qca807x_shared_priv *priv = shared->priv;
|
||||
int val, ret;
|
||||
|
||||
+ /* Make sure PHY follow PHY package mode if enforced */
|
||||
+ if (priv->package_mode != PHY_INTERFACE_MODE_NA &&
|
||||
+ phydev->interface != priv->package_mode)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
phy_lock_mdio_bus(phydev);
|
||||
|
||||
/* Set correct PHY package mode */
|
||||
@@ -718,11 +723,6 @@ static int qca807x_probe(struct phy_devi
|
||||
shared = phydev->shared;
|
||||
shared_priv = shared->priv;
|
||||
|
||||
- /* Make sure PHY follow PHY package mode if enforced */
|
||||
- if (shared_priv->package_mode != PHY_INTERFACE_MODE_NA &&
|
||||
- phydev->interface != shared_priv->package_mode)
|
||||
- return -EINVAL;
|
||||
-
|
||||
priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
|
||||
if (!priv)
|
||||
return -ENOMEM;
|
@ -0,0 +1,205 @@
|
||||
From bdce82e960d1205d118662f575cec39379984e34 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Wed, 31 Jan 2024 03:26:04 +0100
|
||||
Subject: [PATCH] net: mdio: ipq4019: add support for clock-frequency property
|
||||
|
||||
The IPQ4019 MDIO internally divide the clock feed by AHB based on the
|
||||
MDIO_MODE reg. On reset or power up, the default value for the
|
||||
divider is 0xff that reflect the divider set to /256.
|
||||
|
||||
This makes the MDC run at a very low rate, that is, considering AHB is
|
||||
always fixed to 100Mhz, a value of 390KHz.
|
||||
|
||||
This hasn't have been a problem as MDIO wasn't used for time sensitive
|
||||
operation, it is now that on IPQ807x is usually mounted with PHY that
|
||||
requires MDIO to load their firmware (example Aquantia PHY).
|
||||
|
||||
To handle this problem and permit to set the correct designed MDC
|
||||
frequency for the SoC add support for the standard "clock-frequency"
|
||||
property for the MDIO node.
|
||||
|
||||
The divider supports value from /1 to /256 and the common value are to
|
||||
set it to /16 to reflect 6.25Mhz or to /8 on newer platform to reflect
|
||||
12.5Mhz.
|
||||
|
||||
To scan if the requested rate is supported by the divider, loop with
|
||||
each supported divider and stop when the requested rate match the final
|
||||
rate with the current divider. An error is returned if the rate doesn't
|
||||
match any value.
|
||||
|
||||
On MDIO reset, the divider is restored to the requested value to prevent
|
||||
any kind of downclocking caused by the divider reverting to a default
|
||||
value.
|
||||
|
||||
To follow 802.3 spec of 2.5MHz of default value, if divider is set at
|
||||
/256 and "clock-frequency" is not set in DT, assume nobody set the
|
||||
divider and try to find the closest MDC rate to 2.5MHz. (in the case of
|
||||
AHB set to 100MHz, it's 1.5625MHz)
|
||||
|
||||
While at is also document other bits of the MDIO_MODE reg to have a
|
||||
clear idea of what is actually applied there.
|
||||
|
||||
Documentation of some BITs is skipped as they are marked as reserved and
|
||||
their usage is not clear (RES 11:9 GENPHY 16:13 RES1 19:17)
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/mdio/mdio-ipq4019.c | 109 ++++++++++++++++++++++++++++++--
|
||||
1 file changed, 103 insertions(+), 6 deletions(-)
|
||||
|
||||
--- a/drivers/net/mdio/mdio-ipq4019.c
|
||||
+++ b/drivers/net/mdio/mdio-ipq4019.c
|
||||
@@ -14,6 +14,20 @@
|
||||
#include <linux/clk.h>
|
||||
|
||||
#define MDIO_MODE_REG 0x40
|
||||
+#define MDIO_MODE_MDC_MODE BIT(12)
|
||||
+/* 0 = Clause 22, 1 = Clause 45 */
|
||||
+#define MDIO_MODE_C45 BIT(8)
|
||||
+#define MDIO_MODE_DIV_MASK GENMASK(7, 0)
|
||||
+#define MDIO_MODE_DIV(x) FIELD_PREP(MDIO_MODE_DIV_MASK, (x) - 1)
|
||||
+#define MDIO_MODE_DIV_1 0x0
|
||||
+#define MDIO_MODE_DIV_2 0x1
|
||||
+#define MDIO_MODE_DIV_4 0x3
|
||||
+#define MDIO_MODE_DIV_8 0x7
|
||||
+#define MDIO_MODE_DIV_16 0xf
|
||||
+#define MDIO_MODE_DIV_32 0x1f
|
||||
+#define MDIO_MODE_DIV_64 0x3f
|
||||
+#define MDIO_MODE_DIV_128 0x7f
|
||||
+#define MDIO_MODE_DIV_256 0xff
|
||||
#define MDIO_ADDR_REG 0x44
|
||||
#define MDIO_DATA_WRITE_REG 0x48
|
||||
#define MDIO_DATA_READ_REG 0x4c
|
||||
@@ -26,9 +40,6 @@
|
||||
#define MDIO_CMD_ACCESS_CODE_C45_WRITE 1
|
||||
#define MDIO_CMD_ACCESS_CODE_C45_READ 2
|
||||
|
||||
-/* 0 = Clause 22, 1 = Clause 45 */
|
||||
-#define MDIO_MODE_C45 BIT(8)
|
||||
-
|
||||
#define IPQ4019_MDIO_TIMEOUT 10000
|
||||
#define IPQ4019_MDIO_SLEEP 10
|
||||
|
||||
@@ -41,6 +52,7 @@ struct ipq4019_mdio_data {
|
||||
void __iomem *membase;
|
||||
void __iomem *eth_ldo_rdy;
|
||||
struct clk *mdio_clk;
|
||||
+ unsigned int mdc_rate;
|
||||
};
|
||||
|
||||
static int ipq4019_mdio_wait_busy(struct mii_bus *bus)
|
||||
@@ -203,6 +215,38 @@ static int ipq4019_mdio_write_c22(struct
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int ipq4019_mdio_set_div(struct ipq4019_mdio_data *priv)
|
||||
+{
|
||||
+ unsigned long ahb_rate;
|
||||
+ int div;
|
||||
+ u32 val;
|
||||
+
|
||||
+ /* If we don't have a clock for AHB use the fixed value */
|
||||
+ ahb_rate = IPQ_MDIO_CLK_RATE;
|
||||
+ if (priv->mdio_clk)
|
||||
+ ahb_rate = clk_get_rate(priv->mdio_clk);
|
||||
+
|
||||
+ /* MDC rate is ahb_rate/(MDIO_MODE_DIV + 1)
|
||||
+ * While supported, internal documentation doesn't
|
||||
+ * assure correct functionality of the MDIO bus
|
||||
+ * with divider of 1, 2 or 4.
|
||||
+ */
|
||||
+ for (div = 8; div <= 256; div *= 2) {
|
||||
+ /* The requested rate is supported by the div */
|
||||
+ if (priv->mdc_rate == DIV_ROUND_UP(ahb_rate, div)) {
|
||||
+ val = readl(priv->membase + MDIO_MODE_REG);
|
||||
+ val &= ~MDIO_MODE_DIV_MASK;
|
||||
+ val |= MDIO_MODE_DIV(div);
|
||||
+ writel(val, priv->membase + MDIO_MODE_REG);
|
||||
+
|
||||
+ return 0;
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ /* The requested rate is not supported */
|
||||
+ return -EINVAL;
|
||||
+}
|
||||
+
|
||||
static int ipq_mdio_reset(struct mii_bus *bus)
|
||||
{
|
||||
struct ipq4019_mdio_data *priv = bus->priv;
|
||||
@@ -225,10 +269,58 @@ static int ipq_mdio_reset(struct mii_bus
|
||||
return ret;
|
||||
|
||||
ret = clk_prepare_enable(priv->mdio_clk);
|
||||
- if (ret == 0)
|
||||
- mdelay(10);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ mdelay(10);
|
||||
|
||||
- return ret;
|
||||
+ /* Restore MDC rate */
|
||||
+ return ipq4019_mdio_set_div(priv);
|
||||
+}
|
||||
+
|
||||
+static void ipq4019_mdio_select_mdc_rate(struct platform_device *pdev,
|
||||
+ struct ipq4019_mdio_data *priv)
|
||||
+{
|
||||
+ unsigned long ahb_rate;
|
||||
+ int div;
|
||||
+ u32 val;
|
||||
+
|
||||
+ /* MDC rate defined in DT, we don't have to decide a default value */
|
||||
+ if (!of_property_read_u32(pdev->dev.of_node, "clock-frequency",
|
||||
+ &priv->mdc_rate))
|
||||
+ return;
|
||||
+
|
||||
+ /* If we don't have a clock for AHB use the fixed value */
|
||||
+ ahb_rate = IPQ_MDIO_CLK_RATE;
|
||||
+ if (priv->mdio_clk)
|
||||
+ ahb_rate = clk_get_rate(priv->mdio_clk);
|
||||
+
|
||||
+ /* Check what is the current div set */
|
||||
+ val = readl(priv->membase + MDIO_MODE_REG);
|
||||
+ div = FIELD_GET(MDIO_MODE_DIV_MASK, val);
|
||||
+
|
||||
+ /* div is not set to the default value of /256
|
||||
+ * Probably someone changed that (bootloader, other drivers)
|
||||
+ * Keep this and don't overwrite it.
|
||||
+ */
|
||||
+ if (div != MDIO_MODE_DIV_256) {
|
||||
+ priv->mdc_rate = DIV_ROUND_UP(ahb_rate, div + 1);
|
||||
+ return;
|
||||
+ }
|
||||
+
|
||||
+ /* If div is /256 assume nobody have set this value and
|
||||
+ * try to find one MDC rate that is close the 802.3 spec of
|
||||
+ * 2.5MHz
|
||||
+ */
|
||||
+ for (div = 256; div >= 8; div /= 2) {
|
||||
+ /* Stop as soon as we found a divider that
|
||||
+ * reached the closest value to 2.5MHz
|
||||
+ */
|
||||
+ if (DIV_ROUND_UP(ahb_rate, div) > 2500000)
|
||||
+ break;
|
||||
+
|
||||
+ priv->mdc_rate = DIV_ROUND_UP(ahb_rate, div);
|
||||
+ }
|
||||
}
|
||||
|
||||
static int ipq4019_mdio_probe(struct platform_device *pdev)
|
||||
@@ -252,6 +344,11 @@ static int ipq4019_mdio_probe(struct pla
|
||||
if (IS_ERR(priv->mdio_clk))
|
||||
return PTR_ERR(priv->mdio_clk);
|
||||
|
||||
+ ipq4019_mdio_select_mdc_rate(pdev, priv);
|
||||
+ ret = ipq4019_mdio_set_div(priv);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
/* The platform resource is provided on the chipset IPQ5018 */
|
||||
/* This resource is optional */
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
|
@ -0,0 +1,92 @@
|
||||
From 7edce370d87a23e8ed46af5b76a9fef1e341b67b Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 28 Nov 2023 14:59:28 +0100
|
||||
Subject: [PATCH] net: phy: aquantia: drop wrong endianness conversion for addr
|
||||
and CRC
|
||||
|
||||
On further testing on BE target with kernel test robot, it was notice
|
||||
that the endianness conversion for addr and CRC in fw_load_memory was
|
||||
wrong.
|
||||
|
||||
Drop the cpu_to_le32 conversion for addr load as it's not needed.
|
||||
|
||||
Use get_unaligned_le32 instead of get_unaligned for FW data word load to
|
||||
correctly convert data in the correct order to follow system endian.
|
||||
|
||||
Also drop the cpu_to_be32 for CRC calculation as it's wrong and would
|
||||
cause different CRC on BE system.
|
||||
The loaded word is swapped internally and MAILBOX calculates the CRC on
|
||||
the swapped word. To correctly calculate the CRC to be later matched
|
||||
with the one from MAILBOX, use an u8 struct and swap the word there to
|
||||
keep the same order on both LE and BE for crc_ccitt_false function.
|
||||
Also add additional comments on how the CRC verification for the loaded
|
||||
section works.
|
||||
|
||||
CRC is calculated as we load the section and verified with the MAILBOX
|
||||
only after the entire section is loaded to skip additional slowdown by
|
||||
loop the section data again.
|
||||
|
||||
Reported-by: kernel test robot <lkp@intel.com>
|
||||
Closes: https://lore.kernel.org/oe-kbuild-all/202311210414.sEJZjlcD-lkp@intel.com/
|
||||
Fixes: e93984ebc1c8 ("net: phy: aquantia: add firmware load support")
|
||||
Tested-by: Robert Marko <robimarko@gmail.com> # ipq8072 LE device
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Link: https://lore.kernel.org/r/20231128135928.9841-1-ansuelsmth@gmail.com
|
||||
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
|
||||
---
|
||||
drivers/net/phy/aquantia/aquantia_firmware.c | 24 ++++++++++++--------
|
||||
1 file changed, 14 insertions(+), 10 deletions(-)
|
||||
|
||||
--- a/drivers/net/phy/aquantia/aquantia_firmware.c
|
||||
+++ b/drivers/net/phy/aquantia/aquantia_firmware.c
|
||||
@@ -93,9 +93,6 @@ static int aqr_fw_load_memory(struct phy
|
||||
u16 crc = 0, up_crc;
|
||||
size_t pos;
|
||||
|
||||
- /* PHY expect addr in LE */
|
||||
- addr = (__force u32)cpu_to_le32(addr);
|
||||
-
|
||||
phy_write_mmd(phydev, MDIO_MMD_VEND1,
|
||||
VEND1_GLOBAL_MAILBOX_INTERFACE1,
|
||||
VEND1_GLOBAL_MAILBOX_INTERFACE1_CRC_RESET);
|
||||
@@ -110,10 +107,11 @@ static int aqr_fw_load_memory(struct phy
|
||||
* If a firmware that is not word aligned is found, please report upstream.
|
||||
*/
|
||||
for (pos = 0; pos < len; pos += sizeof(u32)) {
|
||||
+ u8 crc_data[4];
|
||||
u32 word;
|
||||
|
||||
/* FW data is always stored in little-endian */
|
||||
- word = get_unaligned((const u32 *)(data + pos));
|
||||
+ word = get_unaligned_le32((const u32 *)(data + pos));
|
||||
|
||||
phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_MAILBOX_INTERFACE5,
|
||||
VEND1_GLOBAL_MAILBOX_INTERFACE5_MSW_DATA(word));
|
||||
@@ -124,15 +122,21 @@ static int aqr_fw_load_memory(struct phy
|
||||
VEND1_GLOBAL_MAILBOX_INTERFACE1_EXECUTE |
|
||||
VEND1_GLOBAL_MAILBOX_INTERFACE1_WRITE);
|
||||
|
||||
- /* calculate CRC as we load data to the mailbox.
|
||||
- * We convert word to big-endian as PHY is BE and mailbox will
|
||||
- * return a BE CRC.
|
||||
+ /* Word is swapped internally and MAILBOX CRC is calculated
|
||||
+ * using big-endian order. Mimic what the PHY does to have a
|
||||
+ * matching CRC...
|
||||
*/
|
||||
- word = (__force u32)cpu_to_be32(word);
|
||||
- crc = crc_ccitt_false(crc, (u8 *)&word, sizeof(word));
|
||||
- }
|
||||
+ crc_data[0] = word >> 24;
|
||||
+ crc_data[1] = word >> 16;
|
||||
+ crc_data[2] = word >> 8;
|
||||
+ crc_data[3] = word;
|
||||
|
||||
+ /* ...calculate CRC as we load data... */
|
||||
+ crc = crc_ccitt_false(crc, crc_data, sizeof(crc_data));
|
||||
+ }
|
||||
+ /* ...gets CRC from MAILBOX after we have loaded the entire section... */
|
||||
up_crc = phy_read_mmd(phydev, MDIO_MMD_VEND1, VEND1_GLOBAL_MAILBOX_INTERFACE2);
|
||||
+ /* ...and make sure it does match our calculated CRC */
|
||||
if (crc != up_crc) {
|
||||
phydev_err(phydev, "CRC mismatch: calculated 0x%04x PHY 0x%04x\n",
|
||||
crc, up_crc);
|
@ -0,0 +1,334 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Thu, 23 Mar 2023 10:24:11 +0100
|
||||
Subject: [PATCH] net: ethernet: mtk_eth_soc: improve keeping track of
|
||||
offloaded flows
|
||||
|
||||
Unify tracking of L2 and L3 flows. Use the generic list field in struct
|
||||
mtk_foe_entry for tracking L2 subflows. Preparation for improving
|
||||
flow accounting support.
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
drivers/net/ethernet/mediatek/mtk_ppe.c | 162 ++++++++++++------------
|
||||
drivers/net/ethernet/mediatek/mtk_ppe.h | 15 +--
|
||||
2 files changed, 86 insertions(+), 91 deletions(-)
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe.c
|
||||
@@ -477,42 +477,43 @@ int mtk_foe_entry_set_queue(struct mtk_e
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int
|
||||
+mtk_flow_entry_match_len(struct mtk_eth *eth, struct mtk_foe_entry *entry)
|
||||
+{
|
||||
+ int type = mtk_get_ib1_pkt_type(eth, entry->ib1);
|
||||
+
|
||||
+ if (type > MTK_PPE_PKT_TYPE_IPV4_DSLITE)
|
||||
+ return offsetof(struct mtk_foe_entry, ipv6._rsv);
|
||||
+ else
|
||||
+ return offsetof(struct mtk_foe_entry, ipv4.ib2);
|
||||
+}
|
||||
+
|
||||
static bool
|
||||
mtk_flow_entry_match(struct mtk_eth *eth, struct mtk_flow_entry *entry,
|
||||
- struct mtk_foe_entry *data)
|
||||
+ struct mtk_foe_entry *data, int len)
|
||||
{
|
||||
- int type, len;
|
||||
-
|
||||
if ((data->ib1 ^ entry->data.ib1) & MTK_FOE_IB1_UDP)
|
||||
return false;
|
||||
|
||||
- type = mtk_get_ib1_pkt_type(eth, entry->data.ib1);
|
||||
- if (type > MTK_PPE_PKT_TYPE_IPV4_DSLITE)
|
||||
- len = offsetof(struct mtk_foe_entry, ipv6._rsv);
|
||||
- else
|
||||
- len = offsetof(struct mtk_foe_entry, ipv4.ib2);
|
||||
-
|
||||
return !memcmp(&entry->data.data, &data->data, len - 4);
|
||||
}
|
||||
|
||||
static void
|
||||
-__mtk_foe_entry_clear(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
+__mtk_foe_entry_clear(struct mtk_ppe *ppe, struct mtk_flow_entry *entry,
|
||||
+ bool set_state)
|
||||
{
|
||||
- struct hlist_head *head;
|
||||
struct hlist_node *tmp;
|
||||
|
||||
if (entry->type == MTK_FLOW_TYPE_L2) {
|
||||
rhashtable_remove_fast(&ppe->l2_flows, &entry->l2_node,
|
||||
mtk_flow_l2_ht_params);
|
||||
|
||||
- head = &entry->l2_flows;
|
||||
- hlist_for_each_entry_safe(entry, tmp, head, l2_data.list)
|
||||
- __mtk_foe_entry_clear(ppe, entry);
|
||||
+ hlist_for_each_entry_safe(entry, tmp, &entry->l2_flows, l2_list)
|
||||
+ __mtk_foe_entry_clear(ppe, entry, set_state);
|
||||
return;
|
||||
}
|
||||
|
||||
- hlist_del_init(&entry->list);
|
||||
- if (entry->hash != 0xffff) {
|
||||
+ if (entry->hash != 0xffff && set_state) {
|
||||
struct mtk_foe_entry *hwe = mtk_foe_get_entry(ppe, entry->hash);
|
||||
|
||||
hwe->ib1 &= ~MTK_FOE_IB1_STATE;
|
||||
@@ -533,7 +534,8 @@ __mtk_foe_entry_clear(struct mtk_ppe *pp
|
||||
if (entry->type != MTK_FLOW_TYPE_L2_SUBFLOW)
|
||||
return;
|
||||
|
||||
- hlist_del_init(&entry->l2_data.list);
|
||||
+ hlist_del_init(&entry->l2_list);
|
||||
+ hlist_del_init(&entry->list);
|
||||
kfree(entry);
|
||||
}
|
||||
|
||||
@@ -549,66 +551,55 @@ static int __mtk_foe_entry_idle_time(str
|
||||
return now - timestamp;
|
||||
}
|
||||
|
||||
+static bool
|
||||
+mtk_flow_entry_update(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
+{
|
||||
+ struct mtk_foe_entry foe = {};
|
||||
+ struct mtk_foe_entry *hwe;
|
||||
+ u16 hash = entry->hash;
|
||||
+ int len;
|
||||
+
|
||||
+ if (hash == 0xffff)
|
||||
+ return false;
|
||||
+
|
||||
+ hwe = mtk_foe_get_entry(ppe, hash);
|
||||
+ len = mtk_flow_entry_match_len(ppe->eth, &entry->data);
|
||||
+ memcpy(&foe, hwe, len);
|
||||
+
|
||||
+ if (!mtk_flow_entry_match(ppe->eth, entry, &foe, len) ||
|
||||
+ FIELD_GET(MTK_FOE_IB1_STATE, foe.ib1) != MTK_FOE_STATE_BIND)
|
||||
+ return false;
|
||||
+
|
||||
+ entry->data.ib1 = foe.ib1;
|
||||
+
|
||||
+ return true;
|
||||
+}
|
||||
+
|
||||
static void
|
||||
mtk_flow_entry_update_l2(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
{
|
||||
u32 ib1_ts_mask = mtk_get_ib1_ts_mask(ppe->eth);
|
||||
struct mtk_flow_entry *cur;
|
||||
- struct mtk_foe_entry *hwe;
|
||||
struct hlist_node *tmp;
|
||||
int idle;
|
||||
|
||||
idle = __mtk_foe_entry_idle_time(ppe, entry->data.ib1);
|
||||
- hlist_for_each_entry_safe(cur, tmp, &entry->l2_flows, l2_data.list) {
|
||||
+ hlist_for_each_entry_safe(cur, tmp, &entry->l2_flows, l2_list) {
|
||||
int cur_idle;
|
||||
- u32 ib1;
|
||||
-
|
||||
- hwe = mtk_foe_get_entry(ppe, cur->hash);
|
||||
- ib1 = READ_ONCE(hwe->ib1);
|
||||
|
||||
- if (FIELD_GET(MTK_FOE_IB1_STATE, ib1) != MTK_FOE_STATE_BIND) {
|
||||
- cur->hash = 0xffff;
|
||||
- __mtk_foe_entry_clear(ppe, cur);
|
||||
+ if (!mtk_flow_entry_update(ppe, cur)) {
|
||||
+ __mtk_foe_entry_clear(ppe, entry, false);
|
||||
continue;
|
||||
}
|
||||
|
||||
- cur_idle = __mtk_foe_entry_idle_time(ppe, ib1);
|
||||
+ cur_idle = __mtk_foe_entry_idle_time(ppe, cur->data.ib1);
|
||||
if (cur_idle >= idle)
|
||||
continue;
|
||||
|
||||
idle = cur_idle;
|
||||
entry->data.ib1 &= ~ib1_ts_mask;
|
||||
- entry->data.ib1 |= hwe->ib1 & ib1_ts_mask;
|
||||
- }
|
||||
-}
|
||||
-
|
||||
-static void
|
||||
-mtk_flow_entry_update(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
-{
|
||||
- struct mtk_foe_entry foe = {};
|
||||
- struct mtk_foe_entry *hwe;
|
||||
-
|
||||
- spin_lock_bh(&ppe_lock);
|
||||
-
|
||||
- if (entry->type == MTK_FLOW_TYPE_L2) {
|
||||
- mtk_flow_entry_update_l2(ppe, entry);
|
||||
- goto out;
|
||||
+ entry->data.ib1 |= cur->data.ib1 & ib1_ts_mask;
|
||||
}
|
||||
-
|
||||
- if (entry->hash == 0xffff)
|
||||
- goto out;
|
||||
-
|
||||
- hwe = mtk_foe_get_entry(ppe, entry->hash);
|
||||
- memcpy(&foe, hwe, ppe->eth->soc->foe_entry_size);
|
||||
- if (!mtk_flow_entry_match(ppe->eth, entry, &foe)) {
|
||||
- entry->hash = 0xffff;
|
||||
- goto out;
|
||||
- }
|
||||
-
|
||||
- entry->data.ib1 = foe.ib1;
|
||||
-
|
||||
-out:
|
||||
- spin_unlock_bh(&ppe_lock);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -651,7 +642,8 @@ __mtk_foe_entry_commit(struct mtk_ppe *p
|
||||
void mtk_foe_entry_clear(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
{
|
||||
spin_lock_bh(&ppe_lock);
|
||||
- __mtk_foe_entry_clear(ppe, entry);
|
||||
+ __mtk_foe_entry_clear(ppe, entry, true);
|
||||
+ hlist_del_init(&entry->list);
|
||||
spin_unlock_bh(&ppe_lock);
|
||||
}
|
||||
|
||||
@@ -698,8 +690,8 @@ mtk_foe_entry_commit_subflow(struct mtk_
|
||||
{
|
||||
const struct mtk_soc_data *soc = ppe->eth->soc;
|
||||
struct mtk_flow_entry *flow_info;
|
||||
- struct mtk_foe_entry foe = {}, *hwe;
|
||||
struct mtk_foe_mac_info *l2;
|
||||
+ struct mtk_foe_entry *hwe;
|
||||
u32 ib1_mask = mtk_get_ib1_pkt_type_mask(ppe->eth) | MTK_FOE_IB1_UDP;
|
||||
int type;
|
||||
|
||||
@@ -707,30 +699,30 @@ mtk_foe_entry_commit_subflow(struct mtk_
|
||||
if (!flow_info)
|
||||
return;
|
||||
|
||||
- flow_info->l2_data.base_flow = entry;
|
||||
flow_info->type = MTK_FLOW_TYPE_L2_SUBFLOW;
|
||||
flow_info->hash = hash;
|
||||
hlist_add_head(&flow_info->list,
|
||||
&ppe->foe_flow[hash / soc->hash_offset]);
|
||||
- hlist_add_head(&flow_info->l2_data.list, &entry->l2_flows);
|
||||
+ hlist_add_head(&flow_info->l2_list, &entry->l2_flows);
|
||||
|
||||
hwe = mtk_foe_get_entry(ppe, hash);
|
||||
- memcpy(&foe, hwe, soc->foe_entry_size);
|
||||
- foe.ib1 &= ib1_mask;
|
||||
- foe.ib1 |= entry->data.ib1 & ~ib1_mask;
|
||||
+ memcpy(&flow_info->data, hwe, soc->foe_entry_size);
|
||||
+ flow_info->data.ib1 &= ib1_mask;
|
||||
+ flow_info->data.ib1 |= entry->data.ib1 & ~ib1_mask;
|
||||
|
||||
- l2 = mtk_foe_entry_l2(ppe->eth, &foe);
|
||||
+ l2 = mtk_foe_entry_l2(ppe->eth, &flow_info->data);
|
||||
memcpy(l2, &entry->data.bridge.l2, sizeof(*l2));
|
||||
|
||||
- type = mtk_get_ib1_pkt_type(ppe->eth, foe.ib1);
|
||||
+ type = mtk_get_ib1_pkt_type(ppe->eth, flow_info->data.ib1);
|
||||
if (type == MTK_PPE_PKT_TYPE_IPV4_HNAPT)
|
||||
- memcpy(&foe.ipv4.new, &foe.ipv4.orig, sizeof(foe.ipv4.new));
|
||||
+ memcpy(&flow_info->data.ipv4.new, &flow_info->data.ipv4.orig,
|
||||
+ sizeof(flow_info->data.ipv4.new));
|
||||
else if (type >= MTK_PPE_PKT_TYPE_IPV6_ROUTE_3T && l2->etype == ETH_P_IP)
|
||||
l2->etype = ETH_P_IPV6;
|
||||
|
||||
- *mtk_foe_entry_ib2(ppe->eth, &foe) = entry->data.bridge.ib2;
|
||||
+ *mtk_foe_entry_ib2(ppe->eth, &flow_info->data) = entry->data.bridge.ib2;
|
||||
|
||||
- __mtk_foe_entry_commit(ppe, &foe, hash);
|
||||
+ __mtk_foe_entry_commit(ppe, &flow_info->data, hash);
|
||||
}
|
||||
|
||||
void __mtk_ppe_check_skb(struct mtk_ppe *ppe, struct sk_buff *skb, u16 hash)
|
||||
@@ -740,9 +732,11 @@ void __mtk_ppe_check_skb(struct mtk_ppe
|
||||
struct mtk_foe_entry *hwe = mtk_foe_get_entry(ppe, hash);
|
||||
struct mtk_flow_entry *entry;
|
||||
struct mtk_foe_bridge key = {};
|
||||
+ struct mtk_foe_entry foe = {};
|
||||
struct hlist_node *n;
|
||||
struct ethhdr *eh;
|
||||
bool found = false;
|
||||
+ int entry_len;
|
||||
u8 *tag;
|
||||
|
||||
spin_lock_bh(&ppe_lock);
|
||||
@@ -750,20 +744,14 @@ void __mtk_ppe_check_skb(struct mtk_ppe
|
||||
if (FIELD_GET(MTK_FOE_IB1_STATE, hwe->ib1) == MTK_FOE_STATE_BIND)
|
||||
goto out;
|
||||
|
||||
- hlist_for_each_entry_safe(entry, n, head, list) {
|
||||
- if (entry->type == MTK_FLOW_TYPE_L2_SUBFLOW) {
|
||||
- if (unlikely(FIELD_GET(MTK_FOE_IB1_STATE, hwe->ib1) ==
|
||||
- MTK_FOE_STATE_BIND))
|
||||
- continue;
|
||||
-
|
||||
- entry->hash = 0xffff;
|
||||
- __mtk_foe_entry_clear(ppe, entry);
|
||||
- continue;
|
||||
- }
|
||||
+ entry_len = mtk_flow_entry_match_len(ppe->eth, hwe);
|
||||
+ memcpy(&foe, hwe, entry_len);
|
||||
|
||||
- if (found || !mtk_flow_entry_match(ppe->eth, entry, hwe)) {
|
||||
+ hlist_for_each_entry_safe(entry, n, head, list) {
|
||||
+ if (found ||
|
||||
+ !mtk_flow_entry_match(ppe->eth, entry, &foe, entry_len)) {
|
||||
if (entry->hash != 0xffff)
|
||||
- entry->hash = 0xffff;
|
||||
+ __mtk_foe_entry_clear(ppe, entry, false);
|
||||
continue;
|
||||
}
|
||||
|
||||
@@ -814,9 +802,17 @@ out:
|
||||
|
||||
int mtk_foe_entry_idle_time(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
{
|
||||
- mtk_flow_entry_update(ppe, entry);
|
||||
+ int idle;
|
||||
+
|
||||
+ spin_lock_bh(&ppe_lock);
|
||||
+ if (entry->type == MTK_FLOW_TYPE_L2)
|
||||
+ mtk_flow_entry_update_l2(ppe, entry);
|
||||
+ else
|
||||
+ mtk_flow_entry_update(ppe, entry);
|
||||
+ idle = __mtk_foe_entry_idle_time(ppe, entry->data.ib1);
|
||||
+ spin_unlock_bh(&ppe_lock);
|
||||
|
||||
- return __mtk_foe_entry_idle_time(ppe, entry->data.ib1);
|
||||
+ return idle;
|
||||
}
|
||||
|
||||
int mtk_ppe_prepare_reset(struct mtk_ppe *ppe)
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe.h
|
||||
@@ -286,7 +286,12 @@ enum {
|
||||
|
||||
struct mtk_flow_entry {
|
||||
union {
|
||||
- struct hlist_node list;
|
||||
+ /* regular flows + L2 subflows */
|
||||
+ struct {
|
||||
+ struct hlist_node list;
|
||||
+ struct hlist_node l2_list;
|
||||
+ };
|
||||
+ /* L2 flows */
|
||||
struct {
|
||||
struct rhash_head l2_node;
|
||||
struct hlist_head l2_flows;
|
||||
@@ -296,13 +301,7 @@ struct mtk_flow_entry {
|
||||
s8 wed_index;
|
||||
u8 ppe_index;
|
||||
u16 hash;
|
||||
- union {
|
||||
- struct mtk_foe_entry data;
|
||||
- struct {
|
||||
- struct mtk_flow_entry *base_flow;
|
||||
- struct hlist_node list;
|
||||
- } l2_data;
|
||||
- };
|
||||
+ struct mtk_foe_entry data;
|
||||
struct rhash_head node;
|
||||
unsigned long cookie;
|
||||
};
|
@ -0,0 +1,343 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Thu, 23 Mar 2023 11:05:22 +0100
|
||||
Subject: [PATCH] net: ethernet: mediatek: fix ppe flow accounting for L2
|
||||
flows
|
||||
|
||||
For L2 flows, the packet/byte counters should report the sum of the
|
||||
counters of their subflows, both current and expired.
|
||||
In order to make this work, change the way that accounting data is tracked.
|
||||
Reset counters when a flow enters bind. Once it expires (or enters unbind),
|
||||
store the last counter value in struct mtk_flow_entry.
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe.c
|
||||
@@ -80,9 +80,9 @@ static int mtk_ppe_mib_wait_busy(struct
|
||||
int ret;
|
||||
u32 val;
|
||||
|
||||
- ret = readl_poll_timeout(ppe->base + MTK_PPE_MIB_SER_CR, val,
|
||||
- !(val & MTK_PPE_MIB_SER_CR_ST),
|
||||
- 20, MTK_PPE_WAIT_TIMEOUT_US);
|
||||
+ ret = readl_poll_timeout_atomic(ppe->base + MTK_PPE_MIB_SER_CR, val,
|
||||
+ !(val & MTK_PPE_MIB_SER_CR_ST),
|
||||
+ 20, MTK_PPE_WAIT_TIMEOUT_US);
|
||||
|
||||
if (ret)
|
||||
dev_err(ppe->dev, "MIB table busy");
|
||||
@@ -90,17 +90,31 @@ static int mtk_ppe_mib_wait_busy(struct
|
||||
return ret;
|
||||
}
|
||||
|
||||
-static int mtk_mib_entry_read(struct mtk_ppe *ppe, u16 index, u64 *bytes, u64 *packets)
|
||||
+static inline struct mtk_foe_accounting *
|
||||
+mtk_ppe_acct_data(struct mtk_ppe *ppe, u16 index)
|
||||
+{
|
||||
+ if (!ppe->acct_table)
|
||||
+ return NULL;
|
||||
+
|
||||
+ return ppe->acct_table + index * sizeof(struct mtk_foe_accounting);
|
||||
+}
|
||||
+
|
||||
+struct mtk_foe_accounting *mtk_ppe_mib_entry_read(struct mtk_ppe *ppe, u16 index)
|
||||
{
|
||||
u32 val, cnt_r0, cnt_r1, cnt_r2;
|
||||
+ struct mtk_foe_accounting *acct;
|
||||
int ret;
|
||||
|
||||
val = FIELD_PREP(MTK_PPE_MIB_SER_CR_ADDR, index) | MTK_PPE_MIB_SER_CR_ST;
|
||||
ppe_w32(ppe, MTK_PPE_MIB_SER_CR, val);
|
||||
|
||||
+ acct = mtk_ppe_acct_data(ppe, index);
|
||||
+ if (!acct)
|
||||
+ return NULL;
|
||||
+
|
||||
ret = mtk_ppe_mib_wait_busy(ppe);
|
||||
if (ret)
|
||||
- return ret;
|
||||
+ return acct;
|
||||
|
||||
cnt_r0 = readl(ppe->base + MTK_PPE_MIB_SER_R0);
|
||||
cnt_r1 = readl(ppe->base + MTK_PPE_MIB_SER_R1);
|
||||
@@ -109,19 +123,19 @@ static int mtk_mib_entry_read(struct mtk
|
||||
if (mtk_is_netsys_v3_or_greater(ppe->eth)) {
|
||||
/* 64 bit for each counter */
|
||||
u32 cnt_r3 = readl(ppe->base + MTK_PPE_MIB_SER_R3);
|
||||
- *bytes = ((u64)cnt_r1 << 32) | cnt_r0;
|
||||
- *packets = ((u64)cnt_r3 << 32) | cnt_r2;
|
||||
+ acct->bytes += ((u64)cnt_r1 << 32) | cnt_r0;
|
||||
+ acct->packets += ((u64)cnt_r3 << 32) | cnt_r2;
|
||||
} else {
|
||||
/* 48 bit byte counter, 40 bit packet counter */
|
||||
u32 byte_cnt_low = FIELD_GET(MTK_PPE_MIB_SER_R0_BYTE_CNT_LOW, cnt_r0);
|
||||
u32 byte_cnt_high = FIELD_GET(MTK_PPE_MIB_SER_R1_BYTE_CNT_HIGH, cnt_r1);
|
||||
u32 pkt_cnt_low = FIELD_GET(MTK_PPE_MIB_SER_R1_PKT_CNT_LOW, cnt_r1);
|
||||
u32 pkt_cnt_high = FIELD_GET(MTK_PPE_MIB_SER_R2_PKT_CNT_HIGH, cnt_r2);
|
||||
- *bytes = ((u64)byte_cnt_high << 32) | byte_cnt_low;
|
||||
- *packets = ((u64)pkt_cnt_high << 16) | pkt_cnt_low;
|
||||
+ acct->bytes += ((u64)byte_cnt_high << 32) | byte_cnt_low;
|
||||
+ acct->packets += ((u64)pkt_cnt_high << 16) | pkt_cnt_low;
|
||||
}
|
||||
|
||||
- return 0;
|
||||
+ return acct;
|
||||
}
|
||||
|
||||
static void mtk_ppe_cache_clear(struct mtk_ppe *ppe)
|
||||
@@ -520,14 +534,6 @@ __mtk_foe_entry_clear(struct mtk_ppe *pp
|
||||
hwe->ib1 |= FIELD_PREP(MTK_FOE_IB1_STATE, MTK_FOE_STATE_INVALID);
|
||||
dma_wmb();
|
||||
mtk_ppe_cache_clear(ppe);
|
||||
-
|
||||
- if (ppe->accounting) {
|
||||
- struct mtk_foe_accounting *acct;
|
||||
-
|
||||
- acct = ppe->acct_table + entry->hash * sizeof(*acct);
|
||||
- acct->packets = 0;
|
||||
- acct->bytes = 0;
|
||||
- }
|
||||
}
|
||||
entry->hash = 0xffff;
|
||||
|
||||
@@ -552,11 +558,14 @@ static int __mtk_foe_entry_idle_time(str
|
||||
}
|
||||
|
||||
static bool
|
||||
-mtk_flow_entry_update(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
+mtk_flow_entry_update(struct mtk_ppe *ppe, struct mtk_flow_entry *entry,
|
||||
+ u64 *packets, u64 *bytes)
|
||||
{
|
||||
+ struct mtk_foe_accounting *acct;
|
||||
struct mtk_foe_entry foe = {};
|
||||
struct mtk_foe_entry *hwe;
|
||||
u16 hash = entry->hash;
|
||||
+ bool ret = false;
|
||||
int len;
|
||||
|
||||
if (hash == 0xffff)
|
||||
@@ -567,18 +576,35 @@ mtk_flow_entry_update(struct mtk_ppe *pp
|
||||
memcpy(&foe, hwe, len);
|
||||
|
||||
if (!mtk_flow_entry_match(ppe->eth, entry, &foe, len) ||
|
||||
- FIELD_GET(MTK_FOE_IB1_STATE, foe.ib1) != MTK_FOE_STATE_BIND)
|
||||
- return false;
|
||||
+ FIELD_GET(MTK_FOE_IB1_STATE, foe.ib1) != MTK_FOE_STATE_BIND) {
|
||||
+ acct = mtk_ppe_acct_data(ppe, hash);
|
||||
+ if (acct) {
|
||||
+ entry->prev_packets += acct->packets;
|
||||
+ entry->prev_bytes += acct->bytes;
|
||||
+ }
|
||||
+
|
||||
+ goto out;
|
||||
+ }
|
||||
|
||||
entry->data.ib1 = foe.ib1;
|
||||
+ acct = mtk_ppe_mib_entry_read(ppe, hash);
|
||||
+ ret = true;
|
||||
+
|
||||
+out:
|
||||
+ if (acct) {
|
||||
+ *packets += acct->packets;
|
||||
+ *bytes += acct->bytes;
|
||||
+ }
|
||||
|
||||
- return true;
|
||||
+ return ret;
|
||||
}
|
||||
|
||||
static void
|
||||
mtk_flow_entry_update_l2(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
{
|
||||
u32 ib1_ts_mask = mtk_get_ib1_ts_mask(ppe->eth);
|
||||
+ u64 *packets = &entry->packets;
|
||||
+ u64 *bytes = &entry->bytes;
|
||||
struct mtk_flow_entry *cur;
|
||||
struct hlist_node *tmp;
|
||||
int idle;
|
||||
@@ -587,7 +613,9 @@ mtk_flow_entry_update_l2(struct mtk_ppe
|
||||
hlist_for_each_entry_safe(cur, tmp, &entry->l2_flows, l2_list) {
|
||||
int cur_idle;
|
||||
|
||||
- if (!mtk_flow_entry_update(ppe, cur)) {
|
||||
+ if (!mtk_flow_entry_update(ppe, cur, packets, bytes)) {
|
||||
+ entry->prev_packets += cur->prev_packets;
|
||||
+ entry->prev_bytes += cur->prev_bytes;
|
||||
__mtk_foe_entry_clear(ppe, entry, false);
|
||||
continue;
|
||||
}
|
||||
@@ -602,10 +630,29 @@ mtk_flow_entry_update_l2(struct mtk_ppe
|
||||
}
|
||||
}
|
||||
|
||||
+void mtk_foe_entry_get_stats(struct mtk_ppe *ppe, struct mtk_flow_entry *entry,
|
||||
+ int *idle)
|
||||
+{
|
||||
+ entry->packets = entry->prev_packets;
|
||||
+ entry->bytes = entry->prev_bytes;
|
||||
+
|
||||
+ spin_lock_bh(&ppe_lock);
|
||||
+
|
||||
+ if (entry->type == MTK_FLOW_TYPE_L2)
|
||||
+ mtk_flow_entry_update_l2(ppe, entry);
|
||||
+ else
|
||||
+ mtk_flow_entry_update(ppe, entry, &entry->packets, &entry->bytes);
|
||||
+
|
||||
+ *idle = __mtk_foe_entry_idle_time(ppe, entry->data.ib1);
|
||||
+
|
||||
+ spin_unlock_bh(&ppe_lock);
|
||||
+}
|
||||
+
|
||||
static void
|
||||
__mtk_foe_entry_commit(struct mtk_ppe *ppe, struct mtk_foe_entry *entry,
|
||||
u16 hash)
|
||||
{
|
||||
+ struct mtk_foe_accounting *acct;
|
||||
struct mtk_eth *eth = ppe->eth;
|
||||
u16 timestamp = mtk_eth_timestamp(eth);
|
||||
struct mtk_foe_entry *hwe;
|
||||
@@ -636,6 +683,12 @@ __mtk_foe_entry_commit(struct mtk_ppe *p
|
||||
|
||||
dma_wmb();
|
||||
|
||||
+ acct = mtk_ppe_mib_entry_read(ppe, hash);
|
||||
+ if (acct) {
|
||||
+ acct->packets = 0;
|
||||
+ acct->bytes = 0;
|
||||
+ }
|
||||
+
|
||||
mtk_ppe_cache_clear(ppe);
|
||||
}
|
||||
|
||||
@@ -800,21 +853,6 @@ out:
|
||||
spin_unlock_bh(&ppe_lock);
|
||||
}
|
||||
|
||||
-int mtk_foe_entry_idle_time(struct mtk_ppe *ppe, struct mtk_flow_entry *entry)
|
||||
-{
|
||||
- int idle;
|
||||
-
|
||||
- spin_lock_bh(&ppe_lock);
|
||||
- if (entry->type == MTK_FLOW_TYPE_L2)
|
||||
- mtk_flow_entry_update_l2(ppe, entry);
|
||||
- else
|
||||
- mtk_flow_entry_update(ppe, entry);
|
||||
- idle = __mtk_foe_entry_idle_time(ppe, entry->data.ib1);
|
||||
- spin_unlock_bh(&ppe_lock);
|
||||
-
|
||||
- return idle;
|
||||
-}
|
||||
-
|
||||
int mtk_ppe_prepare_reset(struct mtk_ppe *ppe)
|
||||
{
|
||||
if (!ppe)
|
||||
@@ -842,32 +880,6 @@ int mtk_ppe_prepare_reset(struct mtk_ppe
|
||||
return mtk_ppe_wait_busy(ppe);
|
||||
}
|
||||
|
||||
-struct mtk_foe_accounting *mtk_foe_entry_get_mib(struct mtk_ppe *ppe, u32 index,
|
||||
- struct mtk_foe_accounting *diff)
|
||||
-{
|
||||
- struct mtk_foe_accounting *acct;
|
||||
- int size = sizeof(struct mtk_foe_accounting);
|
||||
- u64 bytes, packets;
|
||||
-
|
||||
- if (!ppe->accounting)
|
||||
- return NULL;
|
||||
-
|
||||
- if (mtk_mib_entry_read(ppe, index, &bytes, &packets))
|
||||
- return NULL;
|
||||
-
|
||||
- acct = ppe->acct_table + index * size;
|
||||
-
|
||||
- acct->bytes += bytes;
|
||||
- acct->packets += packets;
|
||||
-
|
||||
- if (diff) {
|
||||
- diff->bytes = bytes;
|
||||
- diff->packets = packets;
|
||||
- }
|
||||
-
|
||||
- return acct;
|
||||
-}
|
||||
-
|
||||
struct mtk_ppe *mtk_ppe_init(struct mtk_eth *eth, void __iomem *base, int index)
|
||||
{
|
||||
bool accounting = eth->soc->has_accounting;
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe.h
|
||||
@@ -304,6 +304,8 @@ struct mtk_flow_entry {
|
||||
struct mtk_foe_entry data;
|
||||
struct rhash_head node;
|
||||
unsigned long cookie;
|
||||
+ u64 prev_packets, prev_bytes;
|
||||
+ u64 packets, bytes;
|
||||
};
|
||||
|
||||
struct mtk_mib_entry {
|
||||
@@ -348,6 +350,7 @@ void mtk_ppe_deinit(struct mtk_eth *eth)
|
||||
void mtk_ppe_start(struct mtk_ppe *ppe);
|
||||
int mtk_ppe_stop(struct mtk_ppe *ppe);
|
||||
int mtk_ppe_prepare_reset(struct mtk_ppe *ppe);
|
||||
+struct mtk_foe_accounting *mtk_ppe_mib_entry_read(struct mtk_ppe *ppe, u16 index);
|
||||
|
||||
void __mtk_ppe_check_skb(struct mtk_ppe *ppe, struct sk_buff *skb, u16 hash);
|
||||
|
||||
@@ -396,9 +399,8 @@ int mtk_foe_entry_set_queue(struct mtk_e
|
||||
unsigned int queue);
|
||||
int mtk_foe_entry_commit(struct mtk_ppe *ppe, struct mtk_flow_entry *entry);
|
||||
void mtk_foe_entry_clear(struct mtk_ppe *ppe, struct mtk_flow_entry *entry);
|
||||
-int mtk_foe_entry_idle_time(struct mtk_ppe *ppe, struct mtk_flow_entry *entry);
|
||||
int mtk_ppe_debugfs_init(struct mtk_ppe *ppe, int index);
|
||||
-struct mtk_foe_accounting *mtk_foe_entry_get_mib(struct mtk_ppe *ppe, u32 index,
|
||||
- struct mtk_foe_accounting *diff);
|
||||
+void mtk_foe_entry_get_stats(struct mtk_ppe *ppe, struct mtk_flow_entry *entry,
|
||||
+ int *idle);
|
||||
|
||||
#endif
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe_debugfs.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe_debugfs.c
|
||||
@@ -96,7 +96,7 @@ mtk_ppe_debugfs_foe_show(struct seq_file
|
||||
if (bind && state != MTK_FOE_STATE_BIND)
|
||||
continue;
|
||||
|
||||
- acct = mtk_foe_entry_get_mib(ppe, i, NULL);
|
||||
+ acct = mtk_ppe_mib_entry_read(ppe, i);
|
||||
|
||||
type = mtk_get_ib1_pkt_type(ppe->eth, entry->ib1);
|
||||
seq_printf(m, "%05x %s %7s", i,
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe_offload.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe_offload.c
|
||||
@@ -501,24 +501,21 @@ static int
|
||||
mtk_flow_offload_stats(struct mtk_eth *eth, struct flow_cls_offload *f)
|
||||
{
|
||||
struct mtk_flow_entry *entry;
|
||||
- struct mtk_foe_accounting diff;
|
||||
- u32 idle;
|
||||
+ u64 packets, bytes;
|
||||
+ int idle;
|
||||
|
||||
entry = rhashtable_lookup(ð->flow_table, &f->cookie,
|
||||
mtk_flow_ht_params);
|
||||
if (!entry)
|
||||
return -ENOENT;
|
||||
|
||||
- idle = mtk_foe_entry_idle_time(eth->ppe[entry->ppe_index], entry);
|
||||
+ packets = entry->packets;
|
||||
+ bytes = entry->bytes;
|
||||
+ mtk_foe_entry_get_stats(eth->ppe[entry->ppe_index], entry, &idle);
|
||||
+ f->stats.pkts += entry->packets - packets;
|
||||
+ f->stats.bytes += entry->bytes - bytes;
|
||||
f->stats.lastused = jiffies - idle * HZ;
|
||||
|
||||
- if (entry->hash != 0xFFFF &&
|
||||
- mtk_foe_entry_get_mib(eth->ppe[entry->ppe_index], entry->hash,
|
||||
- &diff)) {
|
||||
- f->stats.pkts += diff.packets;
|
||||
- f->stats.bytes += diff.bytes;
|
||||
- }
|
||||
-
|
||||
return 0;
|
||||
}
|
||||
|
@ -0,0 +1,29 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Tue, 12 Sep 2023 10:22:56 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_eth_soc: rely on mtk_pse_port definitions
|
||||
in mtk_flow_set_output_device
|
||||
|
||||
Similar to ethernet ports, rely on mtk_pse_port definitions for
|
||||
pse wdma ports as well.
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Reviewed-by: Simon Horman <horms@kernel.org>
|
||||
Link: https://lore.kernel.org/r/b86bdb717e963e3246c1dec5f736c810703cf056.1694506814.git.lorenzo@kernel.org
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe_offload.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe_offload.c
|
||||
@@ -196,10 +196,10 @@ mtk_flow_set_output_device(struct mtk_et
|
||||
if (mtk_is_netsys_v2_or_greater(eth)) {
|
||||
switch (info.wdma_idx) {
|
||||
case 0:
|
||||
- pse_port = 8;
|
||||
+ pse_port = PSE_WDMA0_PORT;
|
||||
break;
|
||||
case 1:
|
||||
- pse_port = 9;
|
||||
+ pse_port = PSE_WDMA1_PORT;
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
@ -0,0 +1,26 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Tue, 12 Sep 2023 10:28:00 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: check update_wo_rx_stats in
|
||||
mtk_wed_update_rx_stats()
|
||||
|
||||
Check if update_wo_rx_stats function pointer is properly set in
|
||||
mtk_wed_update_rx_stats routine before accessing it.
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Reviewed-by: Simon Horman <horms@kernel.org>
|
||||
Link: https://lore.kernel.org/r/b0d233386e059bccb59f18f69afb79a7806e5ded.1694507226.git.lorenzo@kernel.org
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
@@ -68,6 +68,9 @@ mtk_wed_update_rx_stats(struct mtk_wed_d
|
||||
struct mtk_wed_wo_rx_stats *stats;
|
||||
int i;
|
||||
|
||||
+ if (!wed->wlan.update_wo_rx_stats)
|
||||
+ return;
|
||||
+
|
||||
if (count * sizeof(*stats) > skb->len - sizeof(u32))
|
||||
return;
|
||||
|
@ -0,0 +1,68 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Wed, 13 Sep 2023 20:42:47 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: do not assume offload callbacks are
|
||||
always set
|
||||
|
||||
Check if wlan.offload_enable and wlan.offload_disable callbacks are set
|
||||
in mtk_wed_flow_add/mtk_wed_flow_remove since mt7996 will not rely
|
||||
on them.
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Reviewed-by: Simon Horman <horms@kernel.org>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -1713,19 +1713,20 @@ mtk_wed_irq_set_mask(struct mtk_wed_devi
|
||||
int mtk_wed_flow_add(int index)
|
||||
{
|
||||
struct mtk_wed_hw *hw = hw_list[index];
|
||||
- int ret;
|
||||
+ int ret = 0;
|
||||
|
||||
- if (!hw || !hw->wed_dev)
|
||||
- return -ENODEV;
|
||||
+ mutex_lock(&hw_lock);
|
||||
|
||||
- if (hw->num_flows) {
|
||||
- hw->num_flows++;
|
||||
- return 0;
|
||||
+ if (!hw || !hw->wed_dev) {
|
||||
+ ret = -ENODEV;
|
||||
+ goto out;
|
||||
}
|
||||
|
||||
- mutex_lock(&hw_lock);
|
||||
- if (!hw->wed_dev) {
|
||||
- ret = -ENODEV;
|
||||
+ if (!hw->wed_dev->wlan.offload_enable)
|
||||
+ goto out;
|
||||
+
|
||||
+ if (hw->num_flows) {
|
||||
+ hw->num_flows++;
|
||||
goto out;
|
||||
}
|
||||
|
||||
@@ -1744,14 +1745,15 @@ void mtk_wed_flow_remove(int index)
|
||||
{
|
||||
struct mtk_wed_hw *hw = hw_list[index];
|
||||
|
||||
- if (!hw)
|
||||
- return;
|
||||
+ mutex_lock(&hw_lock);
|
||||
|
||||
- if (--hw->num_flows)
|
||||
- return;
|
||||
+ if (!hw || !hw->wed_dev)
|
||||
+ goto out;
|
||||
|
||||
- mutex_lock(&hw_lock);
|
||||
- if (!hw->wed_dev)
|
||||
+ if (!hw->wed_dev->wlan.offload_disable)
|
||||
+ goto out;
|
||||
+
|
||||
+ if (--hw->num_flows)
|
||||
goto out;
|
||||
|
||||
hw->wed_dev->wlan.offload_disable(hw->wed_dev);
|
@ -0,0 +1,232 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:05 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: introduce versioning utility routines
|
||||
|
||||
Similar to mtk_eth_soc, introduce the following wed versioning
|
||||
utility routines:
|
||||
- mtk_wed_is_v1
|
||||
- mtk_wed_is_v2
|
||||
|
||||
This is a preliminary patch to introduce WED support for MT7988 SoC
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -278,7 +278,7 @@ mtk_wed_assign(struct mtk_wed_device *de
|
||||
if (!hw->wed_dev)
|
||||
goto out;
|
||||
|
||||
- if (hw->version == 1)
|
||||
+ if (mtk_wed_is_v1(hw))
|
||||
return NULL;
|
||||
|
||||
/* MT7986 WED devices do not have any pcie slot restrictions */
|
||||
@@ -359,7 +359,7 @@ mtk_wed_tx_buffer_alloc(struct mtk_wed_d
|
||||
desc->buf0 = cpu_to_le32(buf_phys);
|
||||
desc->buf1 = cpu_to_le32(buf_phys + txd_size);
|
||||
|
||||
- if (dev->hw->version == 1)
|
||||
+ if (mtk_wed_is_v1(dev->hw))
|
||||
ctrl = FIELD_PREP(MTK_WDMA_DESC_CTRL_LEN0, txd_size) |
|
||||
FIELD_PREP(MTK_WDMA_DESC_CTRL_LEN1,
|
||||
MTK_WED_BUF_SIZE - txd_size) |
|
||||
@@ -498,7 +498,7 @@ mtk_wed_set_ext_int(struct mtk_wed_devic
|
||||
{
|
||||
u32 mask = MTK_WED_EXT_INT_STATUS_ERROR_MASK;
|
||||
|
||||
- if (dev->hw->version == 1)
|
||||
+ if (mtk_wed_is_v1(dev->hw))
|
||||
mask |= MTK_WED_EXT_INT_STATUS_TX_DRV_R_RESP_ERR;
|
||||
else
|
||||
mask |= MTK_WED_EXT_INT_STATUS_RX_FBUF_LO_TH |
|
||||
@@ -577,7 +577,7 @@ mtk_wed_dma_disable(struct mtk_wed_devic
|
||||
MTK_WDMA_GLO_CFG_RX_INFO1_PRERES |
|
||||
MTK_WDMA_GLO_CFG_RX_INFO2_PRERES);
|
||||
|
||||
- if (dev->hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(dev->hw)) {
|
||||
regmap_write(dev->hw->mirror, dev->hw->index * 4, 0);
|
||||
wdma_clr(dev, MTK_WDMA_GLO_CFG,
|
||||
MTK_WDMA_GLO_CFG_RX_INFO3_PRERES);
|
||||
@@ -606,7 +606,7 @@ mtk_wed_stop(struct mtk_wed_device *dev)
|
||||
wdma_w32(dev, MTK_WDMA_INT_GRP2, 0);
|
||||
wed_w32(dev, MTK_WED_WPDMA_INT_MASK, 0);
|
||||
|
||||
- if (dev->hw->version == 1)
|
||||
+ if (mtk_wed_is_v1(dev->hw))
|
||||
return;
|
||||
|
||||
wed_w32(dev, MTK_WED_EXT_INT_MASK1, 0);
|
||||
@@ -625,7 +625,7 @@ mtk_wed_deinit(struct mtk_wed_device *de
|
||||
MTK_WED_CTRL_WED_TX_BM_EN |
|
||||
MTK_WED_CTRL_WED_TX_FREE_AGENT_EN);
|
||||
|
||||
- if (dev->hw->version == 1)
|
||||
+ if (mtk_wed_is_v1(dev->hw))
|
||||
return;
|
||||
|
||||
wed_clr(dev, MTK_WED_CTRL,
|
||||
@@ -731,7 +731,7 @@ mtk_wed_bus_init(struct mtk_wed_device *
|
||||
static void
|
||||
mtk_wed_set_wpdma(struct mtk_wed_device *dev)
|
||||
{
|
||||
- if (dev->hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(dev->hw)) {
|
||||
wed_w32(dev, MTK_WED_WPDMA_CFG_BASE, dev->wlan.wpdma_phys);
|
||||
} else {
|
||||
mtk_wed_bus_init(dev);
|
||||
@@ -762,7 +762,7 @@ mtk_wed_hw_init_early(struct mtk_wed_dev
|
||||
MTK_WED_WDMA_GLO_CFG_IDLE_DMAD_SUPPLY;
|
||||
wed_m32(dev, MTK_WED_WDMA_GLO_CFG, mask, set);
|
||||
|
||||
- if (dev->hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(dev->hw)) {
|
||||
u32 offset = dev->hw->index ? 0x04000400 : 0;
|
||||
|
||||
wdma_set(dev, MTK_WDMA_GLO_CFG,
|
||||
@@ -935,7 +935,7 @@ mtk_wed_hw_init(struct mtk_wed_device *d
|
||||
|
||||
wed_w32(dev, MTK_WED_TX_BM_BUF_LEN, MTK_WED_PKT_SIZE);
|
||||
|
||||
- if (dev->hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(dev->hw)) {
|
||||
wed_w32(dev, MTK_WED_TX_BM_TKID,
|
||||
FIELD_PREP(MTK_WED_TX_BM_TKID_START,
|
||||
dev->wlan.token_start) |
|
||||
@@ -968,7 +968,7 @@ mtk_wed_hw_init(struct mtk_wed_device *d
|
||||
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_TX_BM);
|
||||
|
||||
- if (dev->hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(dev->hw)) {
|
||||
wed_set(dev, MTK_WED_CTRL,
|
||||
MTK_WED_CTRL_WED_TX_BM_EN |
|
||||
MTK_WED_CTRL_WED_TX_FREE_AGENT_EN);
|
||||
@@ -1218,7 +1218,7 @@ mtk_wed_reset_dma(struct mtk_wed_device
|
||||
}
|
||||
|
||||
dev->init_done = false;
|
||||
- if (dev->hw->version == 1)
|
||||
+ if (mtk_wed_is_v1(dev->hw))
|
||||
return;
|
||||
|
||||
if (!busy) {
|
||||
@@ -1344,7 +1344,7 @@ mtk_wed_configure_irq(struct mtk_wed_dev
|
||||
MTK_WED_CTRL_WED_TX_BM_EN |
|
||||
MTK_WED_CTRL_WED_TX_FREE_AGENT_EN);
|
||||
|
||||
- if (dev->hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(dev->hw)) {
|
||||
wed_w32(dev, MTK_WED_PCIE_INT_TRIGGER,
|
||||
MTK_WED_PCIE_INT_TRIGGER_STATUS);
|
||||
|
||||
@@ -1417,7 +1417,7 @@ mtk_wed_dma_enable(struct mtk_wed_device
|
||||
MTK_WDMA_GLO_CFG_RX_INFO1_PRERES |
|
||||
MTK_WDMA_GLO_CFG_RX_INFO2_PRERES);
|
||||
|
||||
- if (dev->hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(dev->hw)) {
|
||||
wdma_set(dev, MTK_WDMA_GLO_CFG,
|
||||
MTK_WDMA_GLO_CFG_RX_INFO3_PRERES);
|
||||
} else {
|
||||
@@ -1466,7 +1466,7 @@ mtk_wed_start(struct mtk_wed_device *dev
|
||||
|
||||
mtk_wed_set_ext_int(dev, true);
|
||||
|
||||
- if (dev->hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(dev->hw)) {
|
||||
u32 val = dev->wlan.wpdma_phys | MTK_PCIE_MIRROR_MAP_EN |
|
||||
FIELD_PREP(MTK_PCIE_MIRROR_MAP_WED_ID,
|
||||
dev->hw->index);
|
||||
@@ -1551,7 +1551,7 @@ mtk_wed_attach(struct mtk_wed_device *de
|
||||
}
|
||||
|
||||
mtk_wed_hw_init_early(dev);
|
||||
- if (hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(hw)) {
|
||||
regmap_update_bits(hw->hifsys, HIFSYS_DMA_AG_MAP,
|
||||
BIT(hw->index), 0);
|
||||
} else {
|
||||
@@ -1619,7 +1619,7 @@ static int
|
||||
mtk_wed_txfree_ring_setup(struct mtk_wed_device *dev, void __iomem *regs)
|
||||
{
|
||||
struct mtk_wed_ring *ring = &dev->txfree_ring;
|
||||
- int i, index = dev->hw->version == 1;
|
||||
+ int i, index = mtk_wed_is_v1(dev->hw);
|
||||
|
||||
/*
|
||||
* For txfree event handling, the same DMA ring is shared between WED
|
||||
@@ -1677,7 +1677,7 @@ mtk_wed_irq_get(struct mtk_wed_device *d
|
||||
{
|
||||
u32 val, ext_mask = MTK_WED_EXT_INT_STATUS_ERROR_MASK;
|
||||
|
||||
- if (dev->hw->version == 1)
|
||||
+ if (mtk_wed_is_v1(dev->hw))
|
||||
ext_mask |= MTK_WED_EXT_INT_STATUS_TX_DRV_R_RESP_ERR;
|
||||
else
|
||||
ext_mask |= MTK_WED_EXT_INT_STATUS_RX_FBUF_LO_TH |
|
||||
@@ -1844,7 +1844,7 @@ mtk_wed_setup_tc(struct mtk_wed_device *
|
||||
{
|
||||
struct mtk_wed_hw *hw = wed->hw;
|
||||
|
||||
- if (hw->version < 2)
|
||||
+ if (mtk_wed_is_v1(hw))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
switch (type) {
|
||||
@@ -1918,9 +1918,9 @@ void mtk_wed_add_hw(struct device_node *
|
||||
hw->wdma = wdma;
|
||||
hw->index = index;
|
||||
hw->irq = irq;
|
||||
- hw->version = mtk_is_netsys_v1(eth) ? 1 : 2;
|
||||
+ hw->version = eth->soc->version;
|
||||
|
||||
- if (hw->version == 1) {
|
||||
+ if (mtk_wed_is_v1(hw)) {
|
||||
hw->mirror = syscon_regmap_lookup_by_phandle(eth_np,
|
||||
"mediatek,pcie-mirror");
|
||||
hw->hifsys = syscon_regmap_lookup_by_phandle(eth_np,
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.h
|
||||
@@ -40,6 +40,16 @@ struct mtk_wdma_info {
|
||||
};
|
||||
|
||||
#ifdef CONFIG_NET_MEDIATEK_SOC_WED
|
||||
+static inline bool mtk_wed_is_v1(struct mtk_wed_hw *hw)
|
||||
+{
|
||||
+ return hw->version == 1;
|
||||
+}
|
||||
+
|
||||
+static inline bool mtk_wed_is_v2(struct mtk_wed_hw *hw)
|
||||
+{
|
||||
+ return hw->version == 2;
|
||||
+}
|
||||
+
|
||||
static inline void
|
||||
wed_w32(struct mtk_wed_device *dev, u32 reg, u32 val)
|
||||
{
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_debugfs.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_debugfs.c
|
||||
@@ -261,7 +261,7 @@ void mtk_wed_hw_add_debugfs(struct mtk_w
|
||||
debugfs_create_u32("regidx", 0600, dir, &hw->debugfs_reg);
|
||||
debugfs_create_file_unsafe("regval", 0600, dir, hw, &fops_regval);
|
||||
debugfs_create_file_unsafe("txinfo", 0400, dir, hw, &wed_txinfo_fops);
|
||||
- if (hw->version != 1)
|
||||
+ if (!mtk_wed_is_v1(hw))
|
||||
debugfs_create_file_unsafe("rxinfo", 0400, dir, hw,
|
||||
&wed_rxinfo_fops);
|
||||
}
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
@@ -207,7 +207,7 @@ int mtk_wed_mcu_msg_update(struct mtk_we
|
||||
{
|
||||
struct mtk_wed_wo *wo = dev->hw->wed_wo;
|
||||
|
||||
- if (dev->hw->version == 1)
|
||||
+ if (mtk_wed_is_v1(dev->hw))
|
||||
return 0;
|
||||
|
||||
if (WARN_ON(!wo))
|
@ -0,0 +1,234 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:06 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: do not configure rx offload if not
|
||||
supported
|
||||
|
||||
Check if rx offload is supported running mtk_wed_get_rx_capa routine
|
||||
before configuring it. This is a preliminary patch to introduce Wireless
|
||||
Ethernet Dispatcher (WED) support for MT7988 SoC.
|
||||
|
||||
Co-developed-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -606,7 +606,7 @@ mtk_wed_stop(struct mtk_wed_device *dev)
|
||||
wdma_w32(dev, MTK_WDMA_INT_GRP2, 0);
|
||||
wed_w32(dev, MTK_WED_WPDMA_INT_MASK, 0);
|
||||
|
||||
- if (mtk_wed_is_v1(dev->hw))
|
||||
+ if (!mtk_wed_get_rx_capa(dev))
|
||||
return;
|
||||
|
||||
wed_w32(dev, MTK_WED_EXT_INT_MASK1, 0);
|
||||
@@ -733,16 +733,21 @@ mtk_wed_set_wpdma(struct mtk_wed_device
|
||||
{
|
||||
if (mtk_wed_is_v1(dev->hw)) {
|
||||
wed_w32(dev, MTK_WED_WPDMA_CFG_BASE, dev->wlan.wpdma_phys);
|
||||
- } else {
|
||||
- mtk_wed_bus_init(dev);
|
||||
-
|
||||
- wed_w32(dev, MTK_WED_WPDMA_CFG_BASE, dev->wlan.wpdma_int);
|
||||
- wed_w32(dev, MTK_WED_WPDMA_CFG_INT_MASK, dev->wlan.wpdma_mask);
|
||||
- wed_w32(dev, MTK_WED_WPDMA_CFG_TX, dev->wlan.wpdma_tx);
|
||||
- wed_w32(dev, MTK_WED_WPDMA_CFG_TX_FREE, dev->wlan.wpdma_txfree);
|
||||
- wed_w32(dev, MTK_WED_WPDMA_RX_GLO_CFG, dev->wlan.wpdma_rx_glo);
|
||||
- wed_w32(dev, MTK_WED_WPDMA_RX_RING, dev->wlan.wpdma_rx);
|
||||
+ return;
|
||||
}
|
||||
+
|
||||
+ mtk_wed_bus_init(dev);
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_CFG_BASE, dev->wlan.wpdma_int);
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_CFG_INT_MASK, dev->wlan.wpdma_mask);
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_CFG_TX, dev->wlan.wpdma_tx);
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_CFG_TX_FREE, dev->wlan.wpdma_txfree);
|
||||
+
|
||||
+ if (!mtk_wed_get_rx_capa(dev))
|
||||
+ return;
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_RX_GLO_CFG, dev->wlan.wpdma_rx_glo);
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_RX_RING, dev->wlan.wpdma_rx);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -974,15 +979,17 @@ mtk_wed_hw_init(struct mtk_wed_device *d
|
||||
MTK_WED_CTRL_WED_TX_FREE_AGENT_EN);
|
||||
} else {
|
||||
wed_clr(dev, MTK_WED_TX_TKID_CTRL, MTK_WED_TX_TKID_CTRL_PAUSE);
|
||||
- /* rx hw init */
|
||||
- wed_w32(dev, MTK_WED_WPDMA_RX_D_RST_IDX,
|
||||
- MTK_WED_WPDMA_RX_D_RST_CRX_IDX |
|
||||
- MTK_WED_WPDMA_RX_D_RST_DRV_IDX);
|
||||
- wed_w32(dev, MTK_WED_WPDMA_RX_D_RST_IDX, 0);
|
||||
-
|
||||
- mtk_wed_rx_buffer_hw_init(dev);
|
||||
- mtk_wed_rro_hw_init(dev);
|
||||
- mtk_wed_route_qm_hw_init(dev);
|
||||
+ if (mtk_wed_get_rx_capa(dev)) {
|
||||
+ /* rx hw init */
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_RX_D_RST_IDX,
|
||||
+ MTK_WED_WPDMA_RX_D_RST_CRX_IDX |
|
||||
+ MTK_WED_WPDMA_RX_D_RST_DRV_IDX);
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_RX_D_RST_IDX, 0);
|
||||
+
|
||||
+ mtk_wed_rx_buffer_hw_init(dev);
|
||||
+ mtk_wed_rro_hw_init(dev);
|
||||
+ mtk_wed_route_qm_hw_init(dev);
|
||||
+ }
|
||||
}
|
||||
|
||||
wed_clr(dev, MTK_WED_TX_BM_CTRL, MTK_WED_TX_BM_CTRL_PAUSE);
|
||||
@@ -1354,8 +1361,6 @@ mtk_wed_configure_irq(struct mtk_wed_dev
|
||||
|
||||
wed_clr(dev, MTK_WED_WDMA_INT_CTRL, wdma_mask);
|
||||
} else {
|
||||
- wdma_mask |= FIELD_PREP(MTK_WDMA_INT_MASK_TX_DONE,
|
||||
- GENMASK(1, 0));
|
||||
/* initail tx interrupt trigger */
|
||||
wed_w32(dev, MTK_WED_WPDMA_INT_CTRL_TX,
|
||||
MTK_WED_WPDMA_INT_CTRL_TX0_DONE_EN |
|
||||
@@ -1374,15 +1379,20 @@ mtk_wed_configure_irq(struct mtk_wed_dev
|
||||
FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_TX_FREE_DONE_TRIG,
|
||||
dev->wlan.txfree_tbit));
|
||||
|
||||
- wed_w32(dev, MTK_WED_WPDMA_INT_CTRL_RX,
|
||||
- MTK_WED_WPDMA_INT_CTRL_RX0_EN |
|
||||
- MTK_WED_WPDMA_INT_CTRL_RX0_CLR |
|
||||
- MTK_WED_WPDMA_INT_CTRL_RX1_EN |
|
||||
- MTK_WED_WPDMA_INT_CTRL_RX1_CLR |
|
||||
- FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RX0_DONE_TRIG,
|
||||
- dev->wlan.rx_tbit[0]) |
|
||||
- FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RX1_DONE_TRIG,
|
||||
- dev->wlan.rx_tbit[1]));
|
||||
+ if (mtk_wed_get_rx_capa(dev)) {
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_INT_CTRL_RX,
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RX0_EN |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RX0_CLR |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RX1_EN |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RX1_CLR |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RX0_DONE_TRIG,
|
||||
+ dev->wlan.rx_tbit[0]) |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RX1_DONE_TRIG,
|
||||
+ dev->wlan.rx_tbit[1]));
|
||||
+
|
||||
+ wdma_mask |= FIELD_PREP(MTK_WDMA_INT_MASK_TX_DONE,
|
||||
+ GENMASK(1, 0));
|
||||
+ }
|
||||
|
||||
wed_w32(dev, MTK_WED_WDMA_INT_CLR, wdma_mask);
|
||||
wed_set(dev, MTK_WED_WDMA_INT_CTRL,
|
||||
@@ -1401,6 +1411,8 @@ mtk_wed_configure_irq(struct mtk_wed_dev
|
||||
static void
|
||||
mtk_wed_dma_enable(struct mtk_wed_device *dev)
|
||||
{
|
||||
+ int i;
|
||||
+
|
||||
wed_set(dev, MTK_WED_WPDMA_INT_CTRL, MTK_WED_WPDMA_INT_CTRL_SUBRT_ADV);
|
||||
|
||||
wed_set(dev, MTK_WED_GLO_CFG,
|
||||
@@ -1420,33 +1432,33 @@ mtk_wed_dma_enable(struct mtk_wed_device
|
||||
if (mtk_wed_is_v1(dev->hw)) {
|
||||
wdma_set(dev, MTK_WDMA_GLO_CFG,
|
||||
MTK_WDMA_GLO_CFG_RX_INFO3_PRERES);
|
||||
- } else {
|
||||
- int i;
|
||||
+ return;
|
||||
+ }
|
||||
|
||||
- wed_set(dev, MTK_WED_WPDMA_CTRL,
|
||||
- MTK_WED_WPDMA_CTRL_SDL1_FIXED);
|
||||
+ wed_set(dev, MTK_WED_WPDMA_CTRL,
|
||||
+ MTK_WED_WPDMA_CTRL_SDL1_FIXED);
|
||||
+ wed_set(dev, MTK_WED_WPDMA_GLO_CFG,
|
||||
+ MTK_WED_WPDMA_GLO_CFG_RX_DRV_R0_PKT_PROC |
|
||||
+ MTK_WED_WPDMA_GLO_CFG_RX_DRV_R0_CRX_SYNC);
|
||||
+ wed_clr(dev, MTK_WED_WPDMA_GLO_CFG,
|
||||
+ MTK_WED_WPDMA_GLO_CFG_TX_TKID_KEEP |
|
||||
+ MTK_WED_WPDMA_GLO_CFG_TX_DMAD_DW3_PREV);
|
||||
|
||||
- wed_set(dev, MTK_WED_WDMA_GLO_CFG,
|
||||
- MTK_WED_WDMA_GLO_CFG_TX_DRV_EN |
|
||||
- MTK_WED_WDMA_GLO_CFG_TX_DDONE_CHK);
|
||||
+ if (!mtk_wed_get_rx_capa(dev))
|
||||
+ return;
|
||||
|
||||
- wed_set(dev, MTK_WED_WPDMA_GLO_CFG,
|
||||
- MTK_WED_WPDMA_GLO_CFG_RX_DRV_R0_PKT_PROC |
|
||||
- MTK_WED_WPDMA_GLO_CFG_RX_DRV_R0_CRX_SYNC);
|
||||
-
|
||||
- wed_clr(dev, MTK_WED_WPDMA_GLO_CFG,
|
||||
- MTK_WED_WPDMA_GLO_CFG_TX_TKID_KEEP |
|
||||
- MTK_WED_WPDMA_GLO_CFG_TX_DMAD_DW3_PREV);
|
||||
+ wed_set(dev, MTK_WED_WDMA_GLO_CFG,
|
||||
+ MTK_WED_WDMA_GLO_CFG_TX_DRV_EN |
|
||||
+ MTK_WED_WDMA_GLO_CFG_TX_DDONE_CHK);
|
||||
|
||||
- wed_set(dev, MTK_WED_WPDMA_RX_D_GLO_CFG,
|
||||
- MTK_WED_WPDMA_RX_D_RX_DRV_EN |
|
||||
- FIELD_PREP(MTK_WED_WPDMA_RX_D_RXD_READ_LEN, 0x18) |
|
||||
- FIELD_PREP(MTK_WED_WPDMA_RX_D_INIT_PHASE_RXEN_SEL,
|
||||
- 0x2));
|
||||
+ wed_set(dev, MTK_WED_WPDMA_RX_D_GLO_CFG,
|
||||
+ MTK_WED_WPDMA_RX_D_RX_DRV_EN |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_RX_D_RXD_READ_LEN, 0x18) |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_RX_D_INIT_PHASE_RXEN_SEL,
|
||||
+ 0x2));
|
||||
|
||||
- for (i = 0; i < MTK_WED_RX_QUEUES; i++)
|
||||
- mtk_wed_check_wfdma_rx_fill(dev, i);
|
||||
- }
|
||||
+ for (i = 0; i < MTK_WED_RX_QUEUES; i++)
|
||||
+ mtk_wed_check_wfdma_rx_fill(dev, i);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -1473,7 +1485,7 @@ mtk_wed_start(struct mtk_wed_device *dev
|
||||
|
||||
val |= BIT(0) | (BIT(1) * !!dev->hw->index);
|
||||
regmap_write(dev->hw->mirror, dev->hw->index * 4, val);
|
||||
- } else {
|
||||
+ } else if (mtk_wed_get_rx_capa(dev)) {
|
||||
/* driver set mid ready and only once */
|
||||
wed_w32(dev, MTK_WED_EXT_INT_MASK1,
|
||||
MTK_WED_EXT_INT_STATUS_WPDMA_MID_RDY);
|
||||
@@ -1485,7 +1497,6 @@ mtk_wed_start(struct mtk_wed_device *dev
|
||||
|
||||
if (mtk_wed_rro_cfg(dev))
|
||||
return;
|
||||
-
|
||||
}
|
||||
|
||||
mtk_wed_set_512_support(dev, dev->wlan.wcid_512);
|
||||
@@ -1551,13 +1562,14 @@ mtk_wed_attach(struct mtk_wed_device *de
|
||||
}
|
||||
|
||||
mtk_wed_hw_init_early(dev);
|
||||
- if (mtk_wed_is_v1(hw)) {
|
||||
+ if (mtk_wed_is_v1(hw))
|
||||
regmap_update_bits(hw->hifsys, HIFSYS_DMA_AG_MAP,
|
||||
BIT(hw->index), 0);
|
||||
- } else {
|
||||
+ else
|
||||
dev->rev_id = wed_r32(dev, MTK_WED_REV_ID);
|
||||
+
|
||||
+ if (mtk_wed_get_rx_capa(dev))
|
||||
ret = mtk_wed_wo_init(hw);
|
||||
- }
|
||||
out:
|
||||
if (ret) {
|
||||
dev_err(dev->hw->dev, "failed to attach wed device\n");
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
@@ -207,7 +207,7 @@ int mtk_wed_mcu_msg_update(struct mtk_we
|
||||
{
|
||||
struct mtk_wed_wo *wo = dev->hw->wed_wo;
|
||||
|
||||
- if (mtk_wed_is_v1(dev->hw))
|
||||
+ if (!mtk_wed_get_rx_capa(dev))
|
||||
return 0;
|
||||
|
||||
if (WARN_ON(!wo))
|
@ -0,0 +1,52 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:07 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: rename mtk_rxbm_desc in
|
||||
mtk_wed_bm_desc
|
||||
|
||||
Rename mtk_rxbm_desc structure in mtk_wed_bm_desc since it will be used
|
||||
even on tx side by MT7988 SoC.
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -422,7 +422,7 @@ free_pagelist:
|
||||
static int
|
||||
mtk_wed_rx_buffer_alloc(struct mtk_wed_device *dev)
|
||||
{
|
||||
- struct mtk_rxbm_desc *desc;
|
||||
+ struct mtk_wed_bm_desc *desc;
|
||||
dma_addr_t desc_phys;
|
||||
|
||||
dev->rx_buf_ring.size = dev->wlan.rx_nbuf;
|
||||
@@ -442,7 +442,7 @@ mtk_wed_rx_buffer_alloc(struct mtk_wed_d
|
||||
static void
|
||||
mtk_wed_free_rx_buffer(struct mtk_wed_device *dev)
|
||||
{
|
||||
- struct mtk_rxbm_desc *desc = dev->rx_buf_ring.desc;
|
||||
+ struct mtk_wed_bm_desc *desc = dev->rx_buf_ring.desc;
|
||||
|
||||
if (!desc)
|
||||
return;
|
||||
--- a/include/linux/soc/mediatek/mtk_wed.h
|
||||
+++ b/include/linux/soc/mediatek/mtk_wed.h
|
||||
@@ -45,7 +45,7 @@ enum mtk_wed_wo_cmd {
|
||||
MTK_WED_WO_CMD_WED_END
|
||||
};
|
||||
|
||||
-struct mtk_rxbm_desc {
|
||||
+struct mtk_wed_bm_desc {
|
||||
__le32 buf0;
|
||||
__le32 token;
|
||||
} __packed __aligned(4);
|
||||
@@ -104,7 +104,7 @@ struct mtk_wed_device {
|
||||
|
||||
struct {
|
||||
int size;
|
||||
- struct mtk_rxbm_desc *desc;
|
||||
+ struct mtk_wed_bm_desc *desc;
|
||||
dma_addr_t desc_phys;
|
||||
} rx_buf_ring;
|
||||
|
@ -0,0 +1,87 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:08 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: introduce mtk_wed_buf structure
|
||||
|
||||
Introduce mtk_wed_buf structure to store both virtual and physical
|
||||
addresses allocated in mtk_wed_tx_buffer_alloc() routine. This is a
|
||||
preliminary patch to add WED support for MT7988 SoC since it relies on a
|
||||
different dma descriptor layout not storing page dma addresses.
|
||||
|
||||
Co-developed-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -300,9 +300,9 @@ out:
|
||||
static int
|
||||
mtk_wed_tx_buffer_alloc(struct mtk_wed_device *dev)
|
||||
{
|
||||
+ struct mtk_wed_buf *page_list;
|
||||
struct mtk_wdma_desc *desc;
|
||||
dma_addr_t desc_phys;
|
||||
- void **page_list;
|
||||
int token = dev->wlan.token_start;
|
||||
int ring_size;
|
||||
int n_pages;
|
||||
@@ -343,7 +343,8 @@ mtk_wed_tx_buffer_alloc(struct mtk_wed_d
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
- page_list[page_idx++] = page;
|
||||
+ page_list[page_idx].p = page;
|
||||
+ page_list[page_idx++].phy_addr = page_phys;
|
||||
dma_sync_single_for_cpu(dev->hw->dev, page_phys, PAGE_SIZE,
|
||||
DMA_BIDIRECTIONAL);
|
||||
|
||||
@@ -387,8 +388,8 @@ mtk_wed_tx_buffer_alloc(struct mtk_wed_d
|
||||
static void
|
||||
mtk_wed_free_tx_buffer(struct mtk_wed_device *dev)
|
||||
{
|
||||
+ struct mtk_wed_buf *page_list = dev->tx_buf_ring.pages;
|
||||
struct mtk_wdma_desc *desc = dev->tx_buf_ring.desc;
|
||||
- void **page_list = dev->tx_buf_ring.pages;
|
||||
int page_idx;
|
||||
int i;
|
||||
|
||||
@@ -400,13 +401,12 @@ mtk_wed_free_tx_buffer(struct mtk_wed_de
|
||||
|
||||
for (i = 0, page_idx = 0; i < dev->tx_buf_ring.size;
|
||||
i += MTK_WED_BUF_PER_PAGE) {
|
||||
- void *page = page_list[page_idx++];
|
||||
- dma_addr_t buf_addr;
|
||||
+ dma_addr_t buf_addr = page_list[page_idx].phy_addr;
|
||||
+ void *page = page_list[page_idx++].p;
|
||||
|
||||
if (!page)
|
||||
break;
|
||||
|
||||
- buf_addr = le32_to_cpu(desc[i].buf0);
|
||||
dma_unmap_page(dev->hw->dev, buf_addr, PAGE_SIZE,
|
||||
DMA_BIDIRECTIONAL);
|
||||
__free_page(page);
|
||||
--- a/include/linux/soc/mediatek/mtk_wed.h
|
||||
+++ b/include/linux/soc/mediatek/mtk_wed.h
|
||||
@@ -76,6 +76,11 @@ struct mtk_wed_wo_rx_stats {
|
||||
__le32 rx_drop_cnt;
|
||||
};
|
||||
|
||||
+struct mtk_wed_buf {
|
||||
+ void *p;
|
||||
+ dma_addr_t phy_addr;
|
||||
+};
|
||||
+
|
||||
struct mtk_wed_device {
|
||||
#ifdef CONFIG_NET_MEDIATEK_SOC_WED
|
||||
const struct mtk_wed_ops *ops;
|
||||
@@ -97,7 +102,7 @@ struct mtk_wed_device {
|
||||
|
||||
struct {
|
||||
int size;
|
||||
- void **pages;
|
||||
+ struct mtk_wed_buf *pages;
|
||||
struct mtk_wdma_desc *desc;
|
||||
dma_addr_t desc_phys;
|
||||
} tx_buf_ring;
|
@ -0,0 +1,88 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:09 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: move mem_region array out of
|
||||
mtk_wed_mcu_load_firmware
|
||||
|
||||
Remove mtk_wed_wo_memory_region boot structure in mtk_wed_wo.
|
||||
This is a preliminary patch to introduce WED support for MT7988 SoC.
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
@@ -16,14 +16,30 @@
|
||||
#include "mtk_wed_wo.h"
|
||||
#include "mtk_wed.h"
|
||||
|
||||
+static struct mtk_wed_wo_memory_region mem_region[] = {
|
||||
+ [MTK_WED_WO_REGION_EMI] = {
|
||||
+ .name = "wo-emi",
|
||||
+ },
|
||||
+ [MTK_WED_WO_REGION_ILM] = {
|
||||
+ .name = "wo-ilm",
|
||||
+ },
|
||||
+ [MTK_WED_WO_REGION_DATA] = {
|
||||
+ .name = "wo-data",
|
||||
+ .shared = true,
|
||||
+ },
|
||||
+ [MTK_WED_WO_REGION_BOOT] = {
|
||||
+ .name = "wo-boot",
|
||||
+ },
|
||||
+};
|
||||
+
|
||||
static u32 wo_r32(struct mtk_wed_wo *wo, u32 reg)
|
||||
{
|
||||
- return readl(wo->boot.addr + reg);
|
||||
+ return readl(mem_region[MTK_WED_WO_REGION_BOOT].addr + reg);
|
||||
}
|
||||
|
||||
static void wo_w32(struct mtk_wed_wo *wo, u32 reg, u32 val)
|
||||
{
|
||||
- writel(val, wo->boot.addr + reg);
|
||||
+ writel(val, mem_region[MTK_WED_WO_REGION_BOOT].addr + reg);
|
||||
}
|
||||
|
||||
static struct sk_buff *
|
||||
@@ -294,18 +310,6 @@ next:
|
||||
static int
|
||||
mtk_wed_mcu_load_firmware(struct mtk_wed_wo *wo)
|
||||
{
|
||||
- static struct mtk_wed_wo_memory_region mem_region[] = {
|
||||
- [MTK_WED_WO_REGION_EMI] = {
|
||||
- .name = "wo-emi",
|
||||
- },
|
||||
- [MTK_WED_WO_REGION_ILM] = {
|
||||
- .name = "wo-ilm",
|
||||
- },
|
||||
- [MTK_WED_WO_REGION_DATA] = {
|
||||
- .name = "wo-data",
|
||||
- .shared = true,
|
||||
- },
|
||||
- };
|
||||
const struct mtk_wed_fw_trailer *trailer;
|
||||
const struct firmware *fw;
|
||||
const char *fw_name;
|
||||
@@ -319,11 +323,6 @@ mtk_wed_mcu_load_firmware(struct mtk_wed
|
||||
return ret;
|
||||
}
|
||||
|
||||
- wo->boot.name = "wo-boot";
|
||||
- ret = mtk_wed_get_memory_region(wo, &wo->boot);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
-
|
||||
/* set dummy cr */
|
||||
wed_w32(wo->hw->wed_dev, MTK_WED_SCR0 + 4 * MTK_WED_DUMMY_CR_FWDL,
|
||||
wo->hw->index + 1);
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_wo.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_wo.h
|
||||
@@ -228,7 +228,6 @@ struct mtk_wed_wo_queue {
|
||||
|
||||
struct mtk_wed_wo {
|
||||
struct mtk_wed_hw *hw;
|
||||
- struct mtk_wed_wo_memory_region boot;
|
||||
|
||||
struct mtk_wed_wo_queue q_tx;
|
||||
struct mtk_wed_wo_queue q_rx;
|
@ -0,0 +1,71 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:10 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: make memory region optional
|
||||
|
||||
Make mtk_wed_wo_memory_region optionals.
|
||||
This is a preliminary patch to introduce Wireless Ethernet Dispatcher
|
||||
support for MT7988 SoC since MT7988 WED fw image will have a different
|
||||
layout.
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_mcu.c
|
||||
@@ -234,19 +234,13 @@ int mtk_wed_mcu_msg_update(struct mtk_we
|
||||
}
|
||||
|
||||
static int
|
||||
-mtk_wed_get_memory_region(struct mtk_wed_wo *wo,
|
||||
+mtk_wed_get_memory_region(struct mtk_wed_hw *hw, int index,
|
||||
struct mtk_wed_wo_memory_region *region)
|
||||
{
|
||||
struct reserved_mem *rmem;
|
||||
struct device_node *np;
|
||||
- int index;
|
||||
|
||||
- index = of_property_match_string(wo->hw->node, "memory-region-names",
|
||||
- region->name);
|
||||
- if (index < 0)
|
||||
- return index;
|
||||
-
|
||||
- np = of_parse_phandle(wo->hw->node, "memory-region", index);
|
||||
+ np = of_parse_phandle(hw->node, "memory-region", index);
|
||||
if (!np)
|
||||
return -ENODEV;
|
||||
|
||||
@@ -258,7 +252,7 @@ mtk_wed_get_memory_region(struct mtk_wed
|
||||
|
||||
region->phy_addr = rmem->base;
|
||||
region->size = rmem->size;
|
||||
- region->addr = devm_ioremap(wo->hw->dev, region->phy_addr, region->size);
|
||||
+ region->addr = devm_ioremap(hw->dev, region->phy_addr, region->size);
|
||||
|
||||
return !region->addr ? -EINVAL : 0;
|
||||
}
|
||||
@@ -271,6 +265,9 @@ mtk_wed_mcu_run_firmware(struct mtk_wed_
|
||||
const struct mtk_wed_fw_trailer *trailer;
|
||||
const struct mtk_wed_fw_region *fw_region;
|
||||
|
||||
+ if (!region->phy_addr || !region->size)
|
||||
+ return 0;
|
||||
+
|
||||
trailer_ptr = fw->data + fw->size - sizeof(*trailer);
|
||||
trailer = (const struct mtk_wed_fw_trailer *)trailer_ptr;
|
||||
region_ptr = trailer_ptr - trailer->num_region * sizeof(*fw_region);
|
||||
@@ -318,7 +315,13 @@ mtk_wed_mcu_load_firmware(struct mtk_wed
|
||||
|
||||
/* load firmware region metadata */
|
||||
for (i = 0; i < ARRAY_SIZE(mem_region); i++) {
|
||||
- ret = mtk_wed_get_memory_region(wo, &mem_region[i]);
|
||||
+ int index = of_property_match_string(wo->hw->node,
|
||||
+ "memory-region-names",
|
||||
+ mem_region[i].name);
|
||||
+ if (index < 0)
|
||||
+ continue;
|
||||
+
|
||||
+ ret = mtk_wed_get_memory_region(wo->hw, index, &mem_region[i]);
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
@ -0,0 +1,217 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:12 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: add mtk_wed_soc_data structure
|
||||
|
||||
Introduce mtk_wed_soc_data utility structure to contain per-SoC
|
||||
definitions.
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -49,6 +49,26 @@ struct mtk_wed_flow_block_priv {
|
||||
struct net_device *dev;
|
||||
};
|
||||
|
||||
+static const struct mtk_wed_soc_data mt7622_data = {
|
||||
+ .regmap = {
|
||||
+ .tx_bm_tkid = 0x088,
|
||||
+ .wpdma_rx_ring0 = 0x770,
|
||||
+ .reset_idx_tx_mask = GENMASK(3, 0),
|
||||
+ .reset_idx_rx_mask = GENMASK(17, 16),
|
||||
+ },
|
||||
+ .wdma_desc_size = sizeof(struct mtk_wdma_desc),
|
||||
+};
|
||||
+
|
||||
+static const struct mtk_wed_soc_data mt7986_data = {
|
||||
+ .regmap = {
|
||||
+ .tx_bm_tkid = 0x0c8,
|
||||
+ .wpdma_rx_ring0 = 0x770,
|
||||
+ .reset_idx_tx_mask = GENMASK(1, 0),
|
||||
+ .reset_idx_rx_mask = GENMASK(7, 6),
|
||||
+ },
|
||||
+ .wdma_desc_size = 2 * sizeof(struct mtk_wdma_desc),
|
||||
+};
|
||||
+
|
||||
static void
|
||||
wed_m32(struct mtk_wed_device *dev, u32 reg, u32 mask, u32 val)
|
||||
{
|
||||
@@ -747,7 +767,7 @@ mtk_wed_set_wpdma(struct mtk_wed_device
|
||||
return;
|
||||
|
||||
wed_w32(dev, MTK_WED_WPDMA_RX_GLO_CFG, dev->wlan.wpdma_rx_glo);
|
||||
- wed_w32(dev, MTK_WED_WPDMA_RX_RING, dev->wlan.wpdma_rx);
|
||||
+ wed_w32(dev, dev->hw->soc->regmap.wpdma_rx_ring0, dev->wlan.wpdma_rx);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -941,22 +961,10 @@ mtk_wed_hw_init(struct mtk_wed_device *d
|
||||
wed_w32(dev, MTK_WED_TX_BM_BUF_LEN, MTK_WED_PKT_SIZE);
|
||||
|
||||
if (mtk_wed_is_v1(dev->hw)) {
|
||||
- wed_w32(dev, MTK_WED_TX_BM_TKID,
|
||||
- FIELD_PREP(MTK_WED_TX_BM_TKID_START,
|
||||
- dev->wlan.token_start) |
|
||||
- FIELD_PREP(MTK_WED_TX_BM_TKID_END,
|
||||
- dev->wlan.token_start +
|
||||
- dev->wlan.nbuf - 1));
|
||||
wed_w32(dev, MTK_WED_TX_BM_DYN_THR,
|
||||
FIELD_PREP(MTK_WED_TX_BM_DYN_THR_LO, 1) |
|
||||
MTK_WED_TX_BM_DYN_THR_HI);
|
||||
} else {
|
||||
- wed_w32(dev, MTK_WED_TX_BM_TKID_V2,
|
||||
- FIELD_PREP(MTK_WED_TX_BM_TKID_START,
|
||||
- dev->wlan.token_start) |
|
||||
- FIELD_PREP(MTK_WED_TX_BM_TKID_END,
|
||||
- dev->wlan.token_start +
|
||||
- dev->wlan.nbuf - 1));
|
||||
wed_w32(dev, MTK_WED_TX_BM_DYN_THR,
|
||||
FIELD_PREP(MTK_WED_TX_BM_DYN_THR_LO_V2, 0) |
|
||||
MTK_WED_TX_BM_DYN_THR_HI_V2);
|
||||
@@ -971,6 +979,11 @@ mtk_wed_hw_init(struct mtk_wed_device *d
|
||||
MTK_WED_TX_TKID_DYN_THR_HI);
|
||||
}
|
||||
|
||||
+ wed_w32(dev, dev->hw->soc->regmap.tx_bm_tkid,
|
||||
+ FIELD_PREP(MTK_WED_TX_BM_TKID_START, dev->wlan.token_start) |
|
||||
+ FIELD_PREP(MTK_WED_TX_BM_TKID_END,
|
||||
+ dev->wlan.token_start + dev->wlan.nbuf - 1));
|
||||
+
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_TX_BM);
|
||||
|
||||
if (mtk_wed_is_v1(dev->hw)) {
|
||||
@@ -1105,13 +1118,8 @@ mtk_wed_rx_reset(struct mtk_wed_device *
|
||||
if (ret) {
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WED_RX_DMA);
|
||||
} else {
|
||||
- struct mtk_eth *eth = dev->hw->eth;
|
||||
-
|
||||
- if (mtk_is_netsys_v2_or_greater(eth))
|
||||
- wed_set(dev, MTK_WED_RESET_IDX,
|
||||
- MTK_WED_RESET_IDX_RX_V2);
|
||||
- else
|
||||
- wed_set(dev, MTK_WED_RESET_IDX, MTK_WED_RESET_IDX_RX);
|
||||
+ wed_set(dev, MTK_WED_RESET_IDX,
|
||||
+ dev->hw->soc->regmap.reset_idx_rx_mask);
|
||||
wed_w32(dev, MTK_WED_RESET_IDX, 0);
|
||||
}
|
||||
|
||||
@@ -1164,7 +1172,8 @@ mtk_wed_reset_dma(struct mtk_wed_device
|
||||
if (busy) {
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WED_TX_DMA);
|
||||
} else {
|
||||
- wed_w32(dev, MTK_WED_RESET_IDX, MTK_WED_RESET_IDX_TX);
|
||||
+ wed_w32(dev, MTK_WED_RESET_IDX,
|
||||
+ dev->hw->soc->regmap.reset_idx_tx_mask);
|
||||
wed_w32(dev, MTK_WED_RESET_IDX, 0);
|
||||
}
|
||||
|
||||
@@ -1256,7 +1265,6 @@ static int
|
||||
mtk_wed_wdma_rx_ring_setup(struct mtk_wed_device *dev, int idx, int size,
|
||||
bool reset)
|
||||
{
|
||||
- u32 desc_size = sizeof(struct mtk_wdma_desc) * dev->hw->version;
|
||||
struct mtk_wed_ring *wdma;
|
||||
|
||||
if (idx >= ARRAY_SIZE(dev->rx_wdma))
|
||||
@@ -1264,7 +1272,7 @@ mtk_wed_wdma_rx_ring_setup(struct mtk_we
|
||||
|
||||
wdma = &dev->rx_wdma[idx];
|
||||
if (!reset && mtk_wed_ring_alloc(dev, wdma, MTK_WED_WDMA_RING_SIZE,
|
||||
- desc_size, true))
|
||||
+ dev->hw->soc->wdma_desc_size, true))
|
||||
return -ENOMEM;
|
||||
|
||||
wdma_w32(dev, MTK_WDMA_RING_RX(idx) + MTK_WED_RING_OFS_BASE,
|
||||
@@ -1285,7 +1293,6 @@ static int
|
||||
mtk_wed_wdma_tx_ring_setup(struct mtk_wed_device *dev, int idx, int size,
|
||||
bool reset)
|
||||
{
|
||||
- u32 desc_size = sizeof(struct mtk_wdma_desc) * dev->hw->version;
|
||||
struct mtk_wed_ring *wdma;
|
||||
|
||||
if (idx >= ARRAY_SIZE(dev->tx_wdma))
|
||||
@@ -1293,7 +1300,7 @@ mtk_wed_wdma_tx_ring_setup(struct mtk_we
|
||||
|
||||
wdma = &dev->tx_wdma[idx];
|
||||
if (!reset && mtk_wed_ring_alloc(dev, wdma, MTK_WED_WDMA_RING_SIZE,
|
||||
- desc_size, true))
|
||||
+ dev->hw->soc->wdma_desc_size, true))
|
||||
return -ENOMEM;
|
||||
|
||||
wdma_w32(dev, MTK_WDMA_RING_TX(idx) + MTK_WED_RING_OFS_BASE,
|
||||
@@ -1932,7 +1939,12 @@ void mtk_wed_add_hw(struct device_node *
|
||||
hw->irq = irq;
|
||||
hw->version = eth->soc->version;
|
||||
|
||||
- if (mtk_wed_is_v1(hw)) {
|
||||
+ switch (hw->version) {
|
||||
+ case 2:
|
||||
+ hw->soc = &mt7986_data;
|
||||
+ break;
|
||||
+ default:
|
||||
+ case 1:
|
||||
hw->mirror = syscon_regmap_lookup_by_phandle(eth_np,
|
||||
"mediatek,pcie-mirror");
|
||||
hw->hifsys = syscon_regmap_lookup_by_phandle(eth_np,
|
||||
@@ -1946,6 +1958,8 @@ void mtk_wed_add_hw(struct device_node *
|
||||
regmap_write(hw->mirror, 0, 0);
|
||||
regmap_write(hw->mirror, 4, 0);
|
||||
}
|
||||
+ hw->soc = &mt7622_data;
|
||||
+ break;
|
||||
}
|
||||
|
||||
mtk_wed_hw_add_debugfs(hw);
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.h
|
||||
@@ -12,7 +12,18 @@
|
||||
struct mtk_eth;
|
||||
struct mtk_wed_wo;
|
||||
|
||||
+struct mtk_wed_soc_data {
|
||||
+ struct {
|
||||
+ u32 tx_bm_tkid;
|
||||
+ u32 wpdma_rx_ring0;
|
||||
+ u32 reset_idx_tx_mask;
|
||||
+ u32 reset_idx_rx_mask;
|
||||
+ } regmap;
|
||||
+ u32 wdma_desc_size;
|
||||
+};
|
||||
+
|
||||
struct mtk_wed_hw {
|
||||
+ const struct mtk_wed_soc_data *soc;
|
||||
struct device_node *node;
|
||||
struct mtk_eth *eth;
|
||||
struct regmap *regs;
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_regs.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_regs.h
|
||||
@@ -100,8 +100,6 @@ struct mtk_wdma_desc {
|
||||
|
||||
#define MTK_WED_TX_BM_BASE 0x084
|
||||
|
||||
-#define MTK_WED_TX_BM_TKID 0x088
|
||||
-#define MTK_WED_TX_BM_TKID_V2 0x0c8
|
||||
#define MTK_WED_TX_BM_TKID_START GENMASK(15, 0)
|
||||
#define MTK_WED_TX_BM_TKID_END GENMASK(31, 16)
|
||||
|
||||
@@ -160,9 +158,6 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_GLO_CFG_RX_2B_OFFSET BIT(31)
|
||||
|
||||
#define MTK_WED_RESET_IDX 0x20c
|
||||
-#define MTK_WED_RESET_IDX_TX GENMASK(3, 0)
|
||||
-#define MTK_WED_RESET_IDX_RX GENMASK(17, 16)
|
||||
-#define MTK_WED_RESET_IDX_RX_V2 GENMASK(7, 6)
|
||||
#define MTK_WED_RESET_WPDMA_IDX_RX GENMASK(31, 30)
|
||||
|
||||
#define MTK_WED_TX_MIB(_n) (0x2a0 + (_n) * 4)
|
||||
@@ -286,7 +281,6 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_WPDMA_RX_D_RST_DRV_IDX GENMASK(25, 24)
|
||||
|
||||
#define MTK_WED_WPDMA_RX_GLO_CFG 0x76c
|
||||
-#define MTK_WED_WPDMA_RX_RING 0x770
|
||||
|
||||
#define MTK_WED_WPDMA_RX_D_MIB(_n) (0x774 + (_n) * 4)
|
||||
#define MTK_WED_WPDMA_RX_D_PROCESSED_MIB(_n) (0x784 + (_n) * 4)
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,95 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:14 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: refactor mtk_wed_check_wfdma_rx_fill
|
||||
routine
|
||||
|
||||
Refactor mtk_wed_check_wfdma_rx_fill() in order to be reused adding HW
|
||||
receive offload support for MT7988 SoC.
|
||||
|
||||
Co-developed-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -586,22 +586,15 @@ mtk_wed_set_512_support(struct mtk_wed_d
|
||||
}
|
||||
}
|
||||
|
||||
-#define MTK_WFMDA_RX_DMA_EN BIT(2)
|
||||
-static void
|
||||
-mtk_wed_check_wfdma_rx_fill(struct mtk_wed_device *dev, int idx)
|
||||
+static int
|
||||
+mtk_wed_check_wfdma_rx_fill(struct mtk_wed_device *dev,
|
||||
+ struct mtk_wed_ring *ring)
|
||||
{
|
||||
- u32 val;
|
||||
int i;
|
||||
|
||||
- if (!(dev->rx_ring[idx].flags & MTK_WED_RING_CONFIGURED))
|
||||
- return; /* queue is not configured by mt76 */
|
||||
-
|
||||
for (i = 0; i < 3; i++) {
|
||||
- u32 cur_idx;
|
||||
+ u32 cur_idx = readl(ring->wpdma + MTK_WED_RING_OFS_CPU_IDX);
|
||||
|
||||
- cur_idx = wed_r32(dev,
|
||||
- MTK_WED_WPDMA_RING_RX_DATA(idx) +
|
||||
- MTK_WED_RING_OFS_CPU_IDX);
|
||||
if (cur_idx == MTK_WED_RX_RING_SIZE - 1)
|
||||
break;
|
||||
|
||||
@@ -610,12 +603,10 @@ mtk_wed_check_wfdma_rx_fill(struct mtk_w
|
||||
|
||||
if (i == 3) {
|
||||
dev_err(dev->hw->dev, "rx dma enable failed\n");
|
||||
- return;
|
||||
+ return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
- val = wifi_r32(dev, dev->wlan.wpdma_rx_glo - dev->wlan.phy_base) |
|
||||
- MTK_WFMDA_RX_DMA_EN;
|
||||
- wifi_w32(dev, dev->wlan.wpdma_rx_glo - dev->wlan.phy_base, val);
|
||||
+ return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -1546,6 +1537,7 @@ mtk_wed_configure_irq(struct mtk_wed_dev
|
||||
wed_w32(dev, MTK_WED_INT_MASK, irq_mask);
|
||||
}
|
||||
|
||||
+#define MTK_WFMDA_RX_DMA_EN BIT(2)
|
||||
static void
|
||||
mtk_wed_dma_enable(struct mtk_wed_device *dev)
|
||||
{
|
||||
@@ -1633,8 +1625,26 @@ mtk_wed_dma_enable(struct mtk_wed_device
|
||||
wdma_set(dev, MTK_WDMA_WRBK_TX_CFG, MTK_WDMA_WRBK_TX_CFG_WRBK_EN);
|
||||
}
|
||||
|
||||
- for (i = 0; i < MTK_WED_RX_QUEUES; i++)
|
||||
- mtk_wed_check_wfdma_rx_fill(dev, i);
|
||||
+ for (i = 0; i < MTK_WED_RX_QUEUES; i++) {
|
||||
+ struct mtk_wed_ring *ring = &dev->rx_ring[i];
|
||||
+ u32 val;
|
||||
+
|
||||
+ if (!(ring->flags & MTK_WED_RING_CONFIGURED))
|
||||
+ continue; /* queue is not configured by mt76 */
|
||||
+
|
||||
+ if (mtk_wed_check_wfdma_rx_fill(dev, ring)) {
|
||||
+ dev_err(dev->hw->dev,
|
||||
+ "rx_ring(%d) dma enable failed\n", i);
|
||||
+ continue;
|
||||
+ }
|
||||
+
|
||||
+ val = wifi_r32(dev,
|
||||
+ dev->wlan.wpdma_rx_glo -
|
||||
+ dev->wlan.phy_base) | MTK_WFMDA_RX_DMA_EN;
|
||||
+ wifi_w32(dev,
|
||||
+ dev->wlan.wpdma_rx_glo - dev->wlan.phy_base,
|
||||
+ val);
|
||||
+ }
|
||||
}
|
||||
|
||||
static void
|
@ -0,0 +1,465 @@
|
||||
From: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Date: Mon, 18 Sep 2023 12:29:15 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: introduce partial AMSDU offload
|
||||
support for MT7988
|
||||
|
||||
Introduce partial AMSDU offload support for MT7988 SoC in order to merge
|
||||
in hw packets belonging to the same AMSDU before passing them to the
|
||||
WLAN nic.
|
||||
|
||||
Co-developed-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe.c
|
||||
@@ -439,7 +439,8 @@ int mtk_foe_entry_set_pppoe(struct mtk_e
|
||||
}
|
||||
|
||||
int mtk_foe_entry_set_wdma(struct mtk_eth *eth, struct mtk_foe_entry *entry,
|
||||
- int wdma_idx, int txq, int bss, int wcid)
|
||||
+ int wdma_idx, int txq, int bss, int wcid,
|
||||
+ bool amsdu_en)
|
||||
{
|
||||
struct mtk_foe_mac_info *l2 = mtk_foe_entry_l2(eth, entry);
|
||||
u32 *ib2 = mtk_foe_entry_ib2(eth, entry);
|
||||
@@ -451,6 +452,7 @@ int mtk_foe_entry_set_wdma(struct mtk_et
|
||||
MTK_FOE_IB2_WDMA_WINFO_V2;
|
||||
l2->w3info = FIELD_PREP(MTK_FOE_WINFO_WCID_V3, wcid) |
|
||||
FIELD_PREP(MTK_FOE_WINFO_BSS_V3, bss);
|
||||
+ l2->amsdu = FIELD_PREP(MTK_FOE_WINFO_AMSDU_EN, amsdu_en);
|
||||
break;
|
||||
case 2:
|
||||
*ib2 &= ~MTK_FOE_IB2_PORT_MG_V2;
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe.h
|
||||
@@ -88,13 +88,13 @@ enum {
|
||||
#define MTK_FOE_WINFO_BSS_V3 GENMASK(23, 16)
|
||||
#define MTK_FOE_WINFO_WCID_V3 GENMASK(15, 0)
|
||||
|
||||
-#define MTK_FOE_WINFO_PAO_USR_INFO GENMASK(15, 0)
|
||||
-#define MTK_FOE_WINFO_PAO_TID GENMASK(19, 16)
|
||||
-#define MTK_FOE_WINFO_PAO_IS_FIXEDRATE BIT(20)
|
||||
-#define MTK_FOE_WINFO_PAO_IS_PRIOR BIT(21)
|
||||
-#define MTK_FOE_WINFO_PAO_IS_SP BIT(22)
|
||||
-#define MTK_FOE_WINFO_PAO_HF BIT(23)
|
||||
-#define MTK_FOE_WINFO_PAO_AMSDU_EN BIT(24)
|
||||
+#define MTK_FOE_WINFO_AMSDU_USR_INFO GENMASK(15, 0)
|
||||
+#define MTK_FOE_WINFO_AMSDU_TID GENMASK(19, 16)
|
||||
+#define MTK_FOE_WINFO_AMSDU_IS_FIXEDRATE BIT(20)
|
||||
+#define MTK_FOE_WINFO_AMSDU_IS_PRIOR BIT(21)
|
||||
+#define MTK_FOE_WINFO_AMSDU_IS_SP BIT(22)
|
||||
+#define MTK_FOE_WINFO_AMSDU_HF BIT(23)
|
||||
+#define MTK_FOE_WINFO_AMSDU_EN BIT(24)
|
||||
|
||||
enum {
|
||||
MTK_FOE_STATE_INVALID,
|
||||
@@ -123,7 +123,7 @@ struct mtk_foe_mac_info {
|
||||
|
||||
/* netsys_v3 */
|
||||
u32 w3info;
|
||||
- u32 wpao;
|
||||
+ u32 amsdu;
|
||||
};
|
||||
|
||||
/* software-only entry type */
|
||||
@@ -394,7 +394,8 @@ int mtk_foe_entry_set_vlan(struct mtk_et
|
||||
int mtk_foe_entry_set_pppoe(struct mtk_eth *eth, struct mtk_foe_entry *entry,
|
||||
int sid);
|
||||
int mtk_foe_entry_set_wdma(struct mtk_eth *eth, struct mtk_foe_entry *entry,
|
||||
- int wdma_idx, int txq, int bss, int wcid);
|
||||
+ int wdma_idx, int txq, int bss, int wcid,
|
||||
+ bool amsdu_en);
|
||||
int mtk_foe_entry_set_queue(struct mtk_eth *eth, struct mtk_foe_entry *entry,
|
||||
unsigned int queue);
|
||||
int mtk_foe_entry_commit(struct mtk_ppe *ppe, struct mtk_flow_entry *entry);
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_ppe_offload.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_ppe_offload.c
|
||||
@@ -111,6 +111,7 @@ mtk_flow_get_wdma_info(struct net_device
|
||||
info->queue = path->mtk_wdma.queue;
|
||||
info->bss = path->mtk_wdma.bss;
|
||||
info->wcid = path->mtk_wdma.wcid;
|
||||
+ info->amsdu = path->mtk_wdma.amsdu;
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -192,7 +193,7 @@ mtk_flow_set_output_device(struct mtk_et
|
||||
|
||||
if (mtk_flow_get_wdma_info(dev, dest_mac, &info) == 0) {
|
||||
mtk_foe_entry_set_wdma(eth, foe, info.wdma_idx, info.queue,
|
||||
- info.bss, info.wcid);
|
||||
+ info.bss, info.wcid, info.amsdu);
|
||||
if (mtk_is_netsys_v2_or_greater(eth)) {
|
||||
switch (info.wdma_idx) {
|
||||
case 0:
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -30,6 +30,8 @@
|
||||
#define MTK_WED_RX_PAGE_BUF_PER_PAGE (PAGE_SIZE / 128)
|
||||
#define MTK_WED_RX_RING_SIZE 1536
|
||||
#define MTK_WED_RX_PG_BM_CNT 8192
|
||||
+#define MTK_WED_AMSDU_BUF_SIZE (PAGE_SIZE << 4)
|
||||
+#define MTK_WED_AMSDU_NPAGES 32
|
||||
|
||||
#define MTK_WED_TX_RING_SIZE 2048
|
||||
#define MTK_WED_WDMA_RING_SIZE 1024
|
||||
@@ -173,6 +175,23 @@ mtk_wdma_rx_reset(struct mtk_wed_device
|
||||
return ret;
|
||||
}
|
||||
|
||||
+static u32
|
||||
+mtk_wed_check_busy(struct mtk_wed_device *dev, u32 reg, u32 mask)
|
||||
+{
|
||||
+ return !!(wed_r32(dev, reg) & mask);
|
||||
+}
|
||||
+
|
||||
+static int
|
||||
+mtk_wed_poll_busy(struct mtk_wed_device *dev, u32 reg, u32 mask)
|
||||
+{
|
||||
+ int sleep = 15000;
|
||||
+ int timeout = 100 * sleep;
|
||||
+ u32 val;
|
||||
+
|
||||
+ return read_poll_timeout(mtk_wed_check_busy, val, !val, sleep,
|
||||
+ timeout, false, dev, reg, mask);
|
||||
+}
|
||||
+
|
||||
static void
|
||||
mtk_wdma_tx_reset(struct mtk_wed_device *dev)
|
||||
{
|
||||
@@ -336,6 +355,118 @@ out:
|
||||
}
|
||||
|
||||
static int
|
||||
+mtk_wed_amsdu_buffer_alloc(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+ struct mtk_wed_hw *hw = dev->hw;
|
||||
+ struct mtk_wed_amsdu *wed_amsdu;
|
||||
+ int i;
|
||||
+
|
||||
+ if (!mtk_wed_is_v3_or_greater(hw))
|
||||
+ return 0;
|
||||
+
|
||||
+ wed_amsdu = devm_kcalloc(hw->dev, MTK_WED_AMSDU_NPAGES,
|
||||
+ sizeof(*wed_amsdu), GFP_KERNEL);
|
||||
+ if (!wed_amsdu)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ for (i = 0; i < MTK_WED_AMSDU_NPAGES; i++) {
|
||||
+ void *ptr;
|
||||
+
|
||||
+ /* each segment is 64K */
|
||||
+ ptr = (void *)__get_free_pages(GFP_KERNEL | __GFP_NOWARN |
|
||||
+ __GFP_ZERO | __GFP_COMP |
|
||||
+ GFP_DMA32,
|
||||
+ get_order(MTK_WED_AMSDU_BUF_SIZE));
|
||||
+ if (!ptr)
|
||||
+ goto error;
|
||||
+
|
||||
+ wed_amsdu[i].txd = ptr;
|
||||
+ wed_amsdu[i].txd_phy = dma_map_single(hw->dev, ptr,
|
||||
+ MTK_WED_AMSDU_BUF_SIZE,
|
||||
+ DMA_TO_DEVICE);
|
||||
+ if (dma_mapping_error(hw->dev, wed_amsdu[i].txd_phy))
|
||||
+ goto error;
|
||||
+ }
|
||||
+ dev->hw->wed_amsdu = wed_amsdu;
|
||||
+
|
||||
+ return 0;
|
||||
+
|
||||
+error:
|
||||
+ for (i--; i >= 0; i--)
|
||||
+ dma_unmap_single(hw->dev, wed_amsdu[i].txd_phy,
|
||||
+ MTK_WED_AMSDU_BUF_SIZE, DMA_TO_DEVICE);
|
||||
+ return -ENOMEM;
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
+mtk_wed_amsdu_free_buffer(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+ struct mtk_wed_amsdu *wed_amsdu = dev->hw->wed_amsdu;
|
||||
+ int i;
|
||||
+
|
||||
+ if (!wed_amsdu)
|
||||
+ return;
|
||||
+
|
||||
+ for (i = 0; i < MTK_WED_AMSDU_NPAGES; i++) {
|
||||
+ dma_unmap_single(dev->hw->dev, wed_amsdu[i].txd_phy,
|
||||
+ MTK_WED_AMSDU_BUF_SIZE, DMA_TO_DEVICE);
|
||||
+ free_pages((unsigned long)wed_amsdu[i].txd,
|
||||
+ get_order(MTK_WED_AMSDU_BUF_SIZE));
|
||||
+ }
|
||||
+}
|
||||
+
|
||||
+static int
|
||||
+mtk_wed_amsdu_init(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+ struct mtk_wed_amsdu *wed_amsdu = dev->hw->wed_amsdu;
|
||||
+ int i, ret;
|
||||
+
|
||||
+ if (!wed_amsdu)
|
||||
+ return 0;
|
||||
+
|
||||
+ for (i = 0; i < MTK_WED_AMSDU_NPAGES; i++)
|
||||
+ wed_w32(dev, MTK_WED_AMSDU_HIFTXD_BASE_L(i),
|
||||
+ wed_amsdu[i].txd_phy);
|
||||
+
|
||||
+ /* init all sta parameter */
|
||||
+ wed_w32(dev, MTK_WED_AMSDU_STA_INFO_INIT, MTK_WED_AMSDU_STA_RMVL |
|
||||
+ MTK_WED_AMSDU_STA_WTBL_HDRT_MODE |
|
||||
+ FIELD_PREP(MTK_WED_AMSDU_STA_MAX_AMSDU_LEN,
|
||||
+ dev->wlan.amsdu_max_len >> 8) |
|
||||
+ FIELD_PREP(MTK_WED_AMSDU_STA_MAX_AMSDU_NUM,
|
||||
+ dev->wlan.amsdu_max_subframes));
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_AMSDU_STA_INFO, MTK_WED_AMSDU_STA_INFO_DO_INIT);
|
||||
+
|
||||
+ ret = mtk_wed_poll_busy(dev, MTK_WED_AMSDU_STA_INFO,
|
||||
+ MTK_WED_AMSDU_STA_INFO_DO_INIT);
|
||||
+ if (ret) {
|
||||
+ dev_err(dev->hw->dev, "amsdu initialization failed\n");
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ /* init partial amsdu offload txd src */
|
||||
+ wed_set(dev, MTK_WED_AMSDU_HIFTXD_CFG,
|
||||
+ FIELD_PREP(MTK_WED_AMSDU_HIFTXD_SRC, dev->hw->index));
|
||||
+
|
||||
+ /* init qmem */
|
||||
+ wed_set(dev, MTK_WED_AMSDU_PSE, MTK_WED_AMSDU_PSE_RESET);
|
||||
+ ret = mtk_wed_poll_busy(dev, MTK_WED_MON_AMSDU_QMEM_STS1, BIT(29));
|
||||
+ if (ret) {
|
||||
+ pr_info("%s: amsdu qmem initialization failed\n", __func__);
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ /* eagle E1 PCIE1 tx ring 22 flow control issue */
|
||||
+ if (dev->wlan.id == 0x7991)
|
||||
+ wed_clr(dev, MTK_WED_AMSDU_FIFO, MTK_WED_AMSDU_IS_PRIOR0_RING);
|
||||
+
|
||||
+ wed_set(dev, MTK_WED_CTRL, MTK_WED_CTRL_TX_AMSDU_EN);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int
|
||||
mtk_wed_tx_buffer_alloc(struct mtk_wed_device *dev)
|
||||
{
|
||||
u32 desc_size = dev->hw->soc->tx_ring_desc_size;
|
||||
@@ -709,6 +840,7 @@ __mtk_wed_detach(struct mtk_wed_device *
|
||||
|
||||
mtk_wdma_rx_reset(dev);
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WED);
|
||||
+ mtk_wed_amsdu_free_buffer(dev);
|
||||
mtk_wed_free_tx_buffer(dev);
|
||||
mtk_wed_free_tx_rings(dev);
|
||||
|
||||
@@ -1129,23 +1261,6 @@ mtk_wed_ring_reset(struct mtk_wed_ring *
|
||||
}
|
||||
}
|
||||
|
||||
-static u32
|
||||
-mtk_wed_check_busy(struct mtk_wed_device *dev, u32 reg, u32 mask)
|
||||
-{
|
||||
- return !!(wed_r32(dev, reg) & mask);
|
||||
-}
|
||||
-
|
||||
-static int
|
||||
-mtk_wed_poll_busy(struct mtk_wed_device *dev, u32 reg, u32 mask)
|
||||
-{
|
||||
- int sleep = 15000;
|
||||
- int timeout = 100 * sleep;
|
||||
- u32 val;
|
||||
-
|
||||
- return read_poll_timeout(mtk_wed_check_busy, val, !val, sleep,
|
||||
- timeout, false, dev, reg, mask);
|
||||
-}
|
||||
-
|
||||
static int
|
||||
mtk_wed_rx_reset(struct mtk_wed_device *dev)
|
||||
{
|
||||
@@ -1692,6 +1807,7 @@ mtk_wed_start(struct mtk_wed_device *dev
|
||||
}
|
||||
|
||||
mtk_wed_set_512_support(dev, dev->wlan.wcid_512);
|
||||
+ mtk_wed_amsdu_init(dev);
|
||||
|
||||
mtk_wed_dma_enable(dev);
|
||||
dev->running = true;
|
||||
@@ -1748,6 +1864,10 @@ mtk_wed_attach(struct mtk_wed_device *de
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
+ ret = mtk_wed_amsdu_buffer_alloc(dev);
|
||||
+ if (ret)
|
||||
+ goto out;
|
||||
+
|
||||
if (mtk_wed_get_rx_capa(dev)) {
|
||||
ret = mtk_wed_rro_alloc(dev);
|
||||
if (ret)
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.h
|
||||
@@ -25,6 +25,11 @@ struct mtk_wed_soc_data {
|
||||
u32 wdma_desc_size;
|
||||
};
|
||||
|
||||
+struct mtk_wed_amsdu {
|
||||
+ void *txd;
|
||||
+ dma_addr_t txd_phy;
|
||||
+};
|
||||
+
|
||||
struct mtk_wed_hw {
|
||||
const struct mtk_wed_soc_data *soc;
|
||||
struct device_node *node;
|
||||
@@ -38,6 +43,7 @@ struct mtk_wed_hw {
|
||||
struct dentry *debugfs_dir;
|
||||
struct mtk_wed_device *wed_dev;
|
||||
struct mtk_wed_wo *wed_wo;
|
||||
+ struct mtk_wed_amsdu *wed_amsdu;
|
||||
u32 pcie_base;
|
||||
u32 debugfs_reg;
|
||||
u32 num_flows;
|
||||
@@ -52,6 +58,7 @@ struct mtk_wdma_info {
|
||||
u8 queue;
|
||||
u16 wcid;
|
||||
u8 bss;
|
||||
+ u8 amsdu;
|
||||
};
|
||||
|
||||
#ifdef CONFIG_NET_MEDIATEK_SOC_WED
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_regs.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_regs.h
|
||||
@@ -672,6 +672,82 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_WOCPU_VIEW_MIOD_BASE 0x8000
|
||||
#define MTK_WED_PCIE_INT_MASK 0x0
|
||||
|
||||
+#define MTK_WED_AMSDU_FIFO 0x1800
|
||||
+#define MTK_WED_AMSDU_IS_PRIOR0_RING BIT(10)
|
||||
+
|
||||
+#define MTK_WED_AMSDU_STA_INFO 0x01810
|
||||
+#define MTK_WED_AMSDU_STA_INFO_DO_INIT BIT(0)
|
||||
+#define MTK_WED_AMSDU_STA_INFO_SET_INIT BIT(1)
|
||||
+
|
||||
+#define MTK_WED_AMSDU_STA_INFO_INIT 0x01814
|
||||
+#define MTK_WED_AMSDU_STA_WTBL_HDRT_MODE BIT(0)
|
||||
+#define MTK_WED_AMSDU_STA_RMVL BIT(1)
|
||||
+#define MTK_WED_AMSDU_STA_MAX_AMSDU_LEN GENMASK(7, 2)
|
||||
+#define MTK_WED_AMSDU_STA_MAX_AMSDU_NUM GENMASK(11, 8)
|
||||
+
|
||||
+#define MTK_WED_AMSDU_HIFTXD_BASE_L(_n) (0x1980 + (_n) * 0x4)
|
||||
+
|
||||
+#define MTK_WED_AMSDU_PSE 0x1910
|
||||
+#define MTK_WED_AMSDU_PSE_RESET BIT(16)
|
||||
+
|
||||
+#define MTK_WED_AMSDU_HIFTXD_CFG 0x1968
|
||||
+#define MTK_WED_AMSDU_HIFTXD_SRC GENMASK(16, 15)
|
||||
+
|
||||
+#define MTK_WED_MON_AMSDU_FIFO_DMAD 0x1a34
|
||||
+
|
||||
+#define MTK_WED_MON_AMSDU_ENG_DMAD(_n) (0x1a80 + (_n) * 0x50)
|
||||
+#define MTK_WED_MON_AMSDU_ENG_QFPL(_n) (0x1a84 + (_n) * 0x50)
|
||||
+#define MTK_WED_MON_AMSDU_ENG_QENI(_n) (0x1a88 + (_n) * 0x50)
|
||||
+#define MTK_WED_MON_AMSDU_ENG_QENO(_n) (0x1a8c + (_n) * 0x50)
|
||||
+#define MTK_WED_MON_AMSDU_ENG_MERG(_n) (0x1a90 + (_n) * 0x50)
|
||||
+
|
||||
+#define MTK_WED_MON_AMSDU_ENG_CNT8(_n) (0x1a94 + (_n) * 0x50)
|
||||
+#define MTK_WED_AMSDU_ENG_MAX_QGPP_CNT GENMASK(10, 0)
|
||||
+#define MTK_WED_AMSDU_ENG_MAX_PL_CNT GENMASK(27, 16)
|
||||
+
|
||||
+#define MTK_WED_MON_AMSDU_ENG_CNT9(_n) (0x1a98 + (_n) * 0x50)
|
||||
+#define MTK_WED_AMSDU_ENG_CUR_ENTRY GENMASK(10, 0)
|
||||
+#define MTK_WED_AMSDU_ENG_MAX_BUF_MERGED GENMASK(20, 16)
|
||||
+#define MTK_WED_AMSDU_ENG_MAX_MSDU_MERGED GENMASK(28, 24)
|
||||
+
|
||||
+#define MTK_WED_MON_AMSDU_QMEM_STS1 0x1e04
|
||||
+
|
||||
+#define MTK_WED_MON_AMSDU_QMEM_CNT(_n) (0x1e0c + (_n) * 0x4)
|
||||
+#define MTK_WED_AMSDU_QMEM_FQ_CNT GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_SP_QCNT GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID0_QCNT GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID1_QCNT GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID2_QCNT GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID3_QCNT GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID4_QCNT GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID5_QCNT GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID6_QCNT GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID7_QCNT GENMASK(11, 0)
|
||||
+
|
||||
+#define MTK_WED_MON_AMSDU_QMEM_PTR(_n) (0x1e20 + (_n) * 0x4)
|
||||
+#define MTK_WED_AMSDU_QMEM_FQ_HEAD GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_SP_QHEAD GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID0_QHEAD GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID1_QHEAD GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID2_QHEAD GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID3_QHEAD GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID4_QHEAD GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID5_QHEAD GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID6_QHEAD GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID7_QHEAD GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_FQ_TAIL GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_SP_QTAIL GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID0_QTAIL GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID1_QTAIL GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID2_QTAIL GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID3_QTAIL GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID4_QTAIL GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID5_QTAIL GENMASK(11, 0)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID6_QTAIL GENMASK(27, 16)
|
||||
+#define MTK_WED_AMSDU_QMEM_TID7_QTAIL GENMASK(11, 0)
|
||||
+
|
||||
+#define MTK_WED_MON_AMSDU_HIFTXD_FETCH_MSDU(_n) (0x1ec4 + (_n) * 0x4)
|
||||
+
|
||||
#define MTK_WED_PCIE_BASE 0x11280000
|
||||
#define MTK_WED_PCIE_BASE0 0x11300000
|
||||
#define MTK_WED_PCIE_BASE1 0x11310000
|
||||
--- a/include/linux/netdevice.h
|
||||
+++ b/include/linux/netdevice.h
|
||||
@@ -917,6 +917,7 @@ struct net_device_path {
|
||||
u8 queue;
|
||||
u16 wcid;
|
||||
u8 bss;
|
||||
+ u8 amsdu;
|
||||
} mtk_wdma;
|
||||
};
|
||||
};
|
||||
--- a/include/linux/soc/mediatek/mtk_wed.h
|
||||
+++ b/include/linux/soc/mediatek/mtk_wed.h
|
||||
@@ -128,6 +128,7 @@ struct mtk_wed_device {
|
||||
enum mtk_wed_bus_tye bus_type;
|
||||
void __iomem *base;
|
||||
u32 phy_base;
|
||||
+ u32 id;
|
||||
|
||||
u32 wpdma_phys;
|
||||
u32 wpdma_int;
|
||||
@@ -146,10 +147,12 @@ struct mtk_wed_device {
|
||||
unsigned int rx_nbuf;
|
||||
unsigned int rx_npkt;
|
||||
unsigned int rx_size;
|
||||
+ unsigned int amsdu_max_len;
|
||||
|
||||
u8 tx_tbit[MTK_WED_TX_QUEUES];
|
||||
u8 rx_tbit[MTK_WED_RX_QUEUES];
|
||||
u8 txfree_tbit;
|
||||
+ u8 amsdu_max_subframes;
|
||||
|
||||
u32 (*init_buf)(void *ptr, dma_addr_t phys, int token_id);
|
||||
int (*offload_enable)(struct mtk_wed_device *wed);
|
||||
@@ -223,6 +226,15 @@ static inline bool mtk_wed_get_rx_capa(s
|
||||
#else
|
||||
return false;
|
||||
#endif
|
||||
+}
|
||||
+
|
||||
+static inline bool mtk_wed_is_amsdu_supported(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+#ifdef CONFIG_NET_MEDIATEK_SOC_WED
|
||||
+ return dev->version == 3;
|
||||
+#else
|
||||
+ return false;
|
||||
+#endif
|
||||
}
|
||||
|
||||
#ifdef CONFIG_NET_MEDIATEK_SOC_WED
|
@ -0,0 +1,483 @@
|
||||
From: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Date: Mon, 18 Sep 2023 12:29:16 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: introduce hw_rro support for MT7988
|
||||
|
||||
MT7988 SoC support 802.11 receive reordering offload in hw while
|
||||
MT7986 SoC implements it through the firmware running on the mcu.
|
||||
|
||||
Co-developed-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -27,7 +27,7 @@
|
||||
#define MTK_WED_BUF_SIZE 2048
|
||||
#define MTK_WED_PAGE_BUF_SIZE 128
|
||||
#define MTK_WED_BUF_PER_PAGE (PAGE_SIZE / 2048)
|
||||
-#define MTK_WED_RX_PAGE_BUF_PER_PAGE (PAGE_SIZE / 128)
|
||||
+#define MTK_WED_RX_BUF_PER_PAGE (PAGE_SIZE / MTK_WED_PAGE_BUF_SIZE)
|
||||
#define MTK_WED_RX_RING_SIZE 1536
|
||||
#define MTK_WED_RX_PG_BM_CNT 8192
|
||||
#define MTK_WED_AMSDU_BUF_SIZE (PAGE_SIZE << 4)
|
||||
@@ -597,6 +597,68 @@ free_pagelist:
|
||||
}
|
||||
|
||||
static int
|
||||
+mtk_wed_hwrro_buffer_alloc(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+ int n_pages = MTK_WED_RX_PG_BM_CNT / MTK_WED_RX_BUF_PER_PAGE;
|
||||
+ struct mtk_wed_buf *page_list;
|
||||
+ struct mtk_wed_bm_desc *desc;
|
||||
+ dma_addr_t desc_phys;
|
||||
+ int i, page_idx = 0;
|
||||
+
|
||||
+ if (!dev->wlan.hw_rro)
|
||||
+ return 0;
|
||||
+
|
||||
+ page_list = kcalloc(n_pages, sizeof(*page_list), GFP_KERNEL);
|
||||
+ if (!page_list)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ dev->hw_rro.size = dev->wlan.rx_nbuf & ~(MTK_WED_BUF_PER_PAGE - 1);
|
||||
+ dev->hw_rro.pages = page_list;
|
||||
+ desc = dma_alloc_coherent(dev->hw->dev,
|
||||
+ dev->wlan.rx_nbuf * sizeof(*desc),
|
||||
+ &desc_phys, GFP_KERNEL);
|
||||
+ if (!desc)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ dev->hw_rro.desc = desc;
|
||||
+ dev->hw_rro.desc_phys = desc_phys;
|
||||
+
|
||||
+ for (i = 0; i < MTK_WED_RX_PG_BM_CNT; i += MTK_WED_RX_BUF_PER_PAGE) {
|
||||
+ dma_addr_t page_phys, buf_phys;
|
||||
+ struct page *page;
|
||||
+ int s;
|
||||
+
|
||||
+ page = __dev_alloc_page(GFP_KERNEL);
|
||||
+ if (!page)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ page_phys = dma_map_page(dev->hw->dev, page, 0, PAGE_SIZE,
|
||||
+ DMA_BIDIRECTIONAL);
|
||||
+ if (dma_mapping_error(dev->hw->dev, page_phys)) {
|
||||
+ __free_page(page);
|
||||
+ return -ENOMEM;
|
||||
+ }
|
||||
+
|
||||
+ page_list[page_idx].p = page;
|
||||
+ page_list[page_idx++].phy_addr = page_phys;
|
||||
+ dma_sync_single_for_cpu(dev->hw->dev, page_phys, PAGE_SIZE,
|
||||
+ DMA_BIDIRECTIONAL);
|
||||
+
|
||||
+ buf_phys = page_phys;
|
||||
+ for (s = 0; s < MTK_WED_RX_BUF_PER_PAGE; s++) {
|
||||
+ desc->buf0 = cpu_to_le32(buf_phys);
|
||||
+ buf_phys += MTK_WED_PAGE_BUF_SIZE;
|
||||
+ desc++;
|
||||
+ }
|
||||
+
|
||||
+ dma_sync_single_for_device(dev->hw->dev, page_phys, PAGE_SIZE,
|
||||
+ DMA_BIDIRECTIONAL);
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static int
|
||||
mtk_wed_rx_buffer_alloc(struct mtk_wed_device *dev)
|
||||
{
|
||||
struct mtk_wed_bm_desc *desc;
|
||||
@@ -613,7 +675,42 @@ mtk_wed_rx_buffer_alloc(struct mtk_wed_d
|
||||
dev->rx_buf_ring.desc_phys = desc_phys;
|
||||
dev->wlan.init_rx_buf(dev, dev->wlan.rx_npkt);
|
||||
|
||||
- return 0;
|
||||
+ return mtk_wed_hwrro_buffer_alloc(dev);
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
+mtk_wed_hwrro_free_buffer(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+ struct mtk_wed_buf *page_list = dev->hw_rro.pages;
|
||||
+ struct mtk_wed_bm_desc *desc = dev->hw_rro.desc;
|
||||
+ int i, page_idx = 0;
|
||||
+
|
||||
+ if (!dev->wlan.hw_rro)
|
||||
+ return;
|
||||
+
|
||||
+ if (!page_list)
|
||||
+ return;
|
||||
+
|
||||
+ if (!desc)
|
||||
+ goto free_pagelist;
|
||||
+
|
||||
+ for (i = 0; i < MTK_WED_RX_PG_BM_CNT; i += MTK_WED_RX_BUF_PER_PAGE) {
|
||||
+ dma_addr_t buf_addr = page_list[page_idx].phy_addr;
|
||||
+ void *page = page_list[page_idx++].p;
|
||||
+
|
||||
+ if (!page)
|
||||
+ break;
|
||||
+
|
||||
+ dma_unmap_page(dev->hw->dev, buf_addr, PAGE_SIZE,
|
||||
+ DMA_BIDIRECTIONAL);
|
||||
+ __free_page(page);
|
||||
+ }
|
||||
+
|
||||
+ dma_free_coherent(dev->hw->dev, dev->hw_rro.size * sizeof(*desc),
|
||||
+ desc, dev->hw_rro.desc_phys);
|
||||
+
|
||||
+free_pagelist:
|
||||
+ kfree(page_list);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -627,6 +724,28 @@ mtk_wed_free_rx_buffer(struct mtk_wed_de
|
||||
dev->wlan.release_rx_buf(dev);
|
||||
dma_free_coherent(dev->hw->dev, dev->rx_buf_ring.size * sizeof(*desc),
|
||||
desc, dev->rx_buf_ring.desc_phys);
|
||||
+
|
||||
+ mtk_wed_hwrro_free_buffer(dev);
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
+mtk_wed_hwrro_init(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+ if (!mtk_wed_get_rx_capa(dev) || !dev->wlan.hw_rro)
|
||||
+ return;
|
||||
+
|
||||
+ wed_set(dev, MTK_WED_RRO_PG_BM_RX_DMAM,
|
||||
+ FIELD_PREP(MTK_WED_RRO_PG_BM_RX_SDL0, 128));
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_RRO_PG_BM_BASE, dev->hw_rro.desc_phys);
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_RRO_PG_BM_INIT_PTR,
|
||||
+ MTK_WED_RRO_PG_BM_INIT_SW_TAIL_IDX |
|
||||
+ FIELD_PREP(MTK_WED_RRO_PG_BM_SW_TAIL_IDX,
|
||||
+ MTK_WED_RX_PG_BM_CNT));
|
||||
+
|
||||
+ /* enable rx_page_bm to fetch dmad */
|
||||
+ wed_set(dev, MTK_WED_CTRL, MTK_WED_CTRL_WED_RX_PG_BM_EN);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -640,6 +759,8 @@ mtk_wed_rx_buffer_hw_init(struct mtk_wed
|
||||
wed_w32(dev, MTK_WED_RX_BM_DYN_ALLOC_TH,
|
||||
FIELD_PREP(MTK_WED_RX_BM_DYN_ALLOC_TH_H, 0xffff));
|
||||
wed_set(dev, MTK_WED_CTRL, MTK_WED_CTRL_WED_RX_BM_EN);
|
||||
+
|
||||
+ mtk_wed_hwrro_init(dev);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -935,6 +1056,8 @@ mtk_wed_bus_init(struct mtk_wed_device *
|
||||
static void
|
||||
mtk_wed_set_wpdma(struct mtk_wed_device *dev)
|
||||
{
|
||||
+ int i;
|
||||
+
|
||||
if (mtk_wed_is_v1(dev->hw)) {
|
||||
wed_w32(dev, MTK_WED_WPDMA_CFG_BASE, dev->wlan.wpdma_phys);
|
||||
return;
|
||||
@@ -952,6 +1075,15 @@ mtk_wed_set_wpdma(struct mtk_wed_device
|
||||
|
||||
wed_w32(dev, MTK_WED_WPDMA_RX_GLO_CFG, dev->wlan.wpdma_rx_glo);
|
||||
wed_w32(dev, dev->hw->soc->regmap.wpdma_rx_ring0, dev->wlan.wpdma_rx);
|
||||
+
|
||||
+ if (!dev->wlan.hw_rro)
|
||||
+ return;
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_RRO_RX_D_CFG(0), dev->wlan.wpdma_rx_rro[0]);
|
||||
+ wed_w32(dev, MTK_WED_RRO_RX_D_CFG(1), dev->wlan.wpdma_rx_rro[1]);
|
||||
+ for (i = 0; i < MTK_WED_RX_PAGE_QUEUES; i++)
|
||||
+ wed_w32(dev, MTK_WED_RRO_MSDU_PG_RING_CFG(i),
|
||||
+ dev->wlan.wpdma_rx_pg + i * 0x10);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -1763,6 +1895,165 @@ mtk_wed_dma_enable(struct mtk_wed_device
|
||||
}
|
||||
|
||||
static void
|
||||
+mtk_wed_start_hw_rro(struct mtk_wed_device *dev, u32 irq_mask, bool reset)
|
||||
+{
|
||||
+ int i;
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_INT_MASK, irq_mask);
|
||||
+ wed_w32(dev, MTK_WED_INT_MASK, irq_mask);
|
||||
+
|
||||
+ if (!mtk_wed_get_rx_capa(dev) || !dev->wlan.hw_rro)
|
||||
+ return;
|
||||
+
|
||||
+ wed_set(dev, MTK_WED_RRO_RX_D_CFG(2), MTK_WED_RRO_MSDU_PG_DRV_CLR);
|
||||
+ wed_w32(dev, MTK_WED_RRO_MSDU_PG_RING2_CFG,
|
||||
+ MTK_WED_RRO_MSDU_PG_DRV_CLR);
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_INT_CTRL_RRO_RX,
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_RX0_EN |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_RX0_CLR |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_RX1_EN |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_RX1_CLR |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RRO_RX0_DONE_TRIG,
|
||||
+ dev->wlan.rro_rx_tbit[0]) |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RRO_RX1_DONE_TRIG,
|
||||
+ dev->wlan.rro_rx_tbit[1]));
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_INT_CTRL_RRO_MSDU_PG,
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_PG0_EN |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_PG0_CLR |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_PG1_EN |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_PG1_CLR |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_PG2_EN |
|
||||
+ MTK_WED_WPDMA_INT_CTRL_RRO_PG2_CLR |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RRO_PG0_DONE_TRIG,
|
||||
+ dev->wlan.rx_pg_tbit[0]) |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RRO_PG1_DONE_TRIG,
|
||||
+ dev->wlan.rx_pg_tbit[1]) |
|
||||
+ FIELD_PREP(MTK_WED_WPDMA_INT_CTRL_RRO_PG2_DONE_TRIG,
|
||||
+ dev->wlan.rx_pg_tbit[2]));
|
||||
+
|
||||
+ /* RRO_MSDU_PG_RING2_CFG1_FLD_DRV_EN should be enabled after
|
||||
+ * WM FWDL completed, otherwise RRO_MSDU_PG ring may broken
|
||||
+ */
|
||||
+ wed_set(dev, MTK_WED_RRO_MSDU_PG_RING2_CFG,
|
||||
+ MTK_WED_RRO_MSDU_PG_DRV_EN);
|
||||
+
|
||||
+ for (i = 0; i < MTK_WED_RX_QUEUES; i++) {
|
||||
+ struct mtk_wed_ring *ring = &dev->rx_rro_ring[i];
|
||||
+
|
||||
+ if (!(ring->flags & MTK_WED_RING_CONFIGURED))
|
||||
+ continue;
|
||||
+
|
||||
+ if (mtk_wed_check_wfdma_rx_fill(dev, ring))
|
||||
+ dev_err(dev->hw->dev,
|
||||
+ "rx_rro_ring(%d) initialization failed\n", i);
|
||||
+ }
|
||||
+
|
||||
+ for (i = 0; i < MTK_WED_RX_PAGE_QUEUES; i++) {
|
||||
+ struct mtk_wed_ring *ring = &dev->rx_page_ring[i];
|
||||
+
|
||||
+ if (!(ring->flags & MTK_WED_RING_CONFIGURED))
|
||||
+ continue;
|
||||
+
|
||||
+ if (mtk_wed_check_wfdma_rx_fill(dev, ring))
|
||||
+ dev_err(dev->hw->dev,
|
||||
+ "rx_page_ring(%d) initialization failed\n", i);
|
||||
+ }
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
+mtk_wed_rro_rx_ring_setup(struct mtk_wed_device *dev, int idx,
|
||||
+ void __iomem *regs)
|
||||
+{
|
||||
+ struct mtk_wed_ring *ring = &dev->rx_rro_ring[idx];
|
||||
+
|
||||
+ ring->wpdma = regs;
|
||||
+ wed_w32(dev, MTK_WED_RRO_RX_D_RX(idx) + MTK_WED_RING_OFS_BASE,
|
||||
+ readl(regs));
|
||||
+ wed_w32(dev, MTK_WED_RRO_RX_D_RX(idx) + MTK_WED_RING_OFS_COUNT,
|
||||
+ readl(regs + MTK_WED_RING_OFS_COUNT));
|
||||
+ ring->flags |= MTK_WED_RING_CONFIGURED;
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
+mtk_wed_msdu_pg_rx_ring_setup(struct mtk_wed_device *dev, int idx, void __iomem *regs)
|
||||
+{
|
||||
+ struct mtk_wed_ring *ring = &dev->rx_page_ring[idx];
|
||||
+
|
||||
+ ring->wpdma = regs;
|
||||
+ wed_w32(dev, MTK_WED_RRO_MSDU_PG_CTRL0(idx) + MTK_WED_RING_OFS_BASE,
|
||||
+ readl(regs));
|
||||
+ wed_w32(dev, MTK_WED_RRO_MSDU_PG_CTRL0(idx) + MTK_WED_RING_OFS_COUNT,
|
||||
+ readl(regs + MTK_WED_RING_OFS_COUNT));
|
||||
+ ring->flags |= MTK_WED_RING_CONFIGURED;
|
||||
+}
|
||||
+
|
||||
+static int
|
||||
+mtk_wed_ind_rx_ring_setup(struct mtk_wed_device *dev, void __iomem *regs)
|
||||
+{
|
||||
+ struct mtk_wed_ring *ring = &dev->ind_cmd_ring;
|
||||
+ u32 val = readl(regs + MTK_WED_RING_OFS_COUNT);
|
||||
+ int i, count = 0;
|
||||
+
|
||||
+ ring->wpdma = regs;
|
||||
+ wed_w32(dev, MTK_WED_IND_CMD_RX_CTRL1 + MTK_WED_RING_OFS_BASE,
|
||||
+ readl(regs) & 0xfffffff0);
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_IND_CMD_RX_CTRL1 + MTK_WED_RING_OFS_COUNT,
|
||||
+ readl(regs + MTK_WED_RING_OFS_COUNT));
|
||||
+
|
||||
+ /* ack sn cr */
|
||||
+ wed_w32(dev, MTK_WED_RRO_CFG0, dev->wlan.phy_base +
|
||||
+ dev->wlan.ind_cmd.ack_sn_addr);
|
||||
+ wed_w32(dev, MTK_WED_RRO_CFG1,
|
||||
+ FIELD_PREP(MTK_WED_RRO_CFG1_MAX_WIN_SZ,
|
||||
+ dev->wlan.ind_cmd.win_size) |
|
||||
+ FIELD_PREP(MTK_WED_RRO_CFG1_PARTICL_SE_ID,
|
||||
+ dev->wlan.ind_cmd.particular_sid));
|
||||
+
|
||||
+ /* particular session addr element */
|
||||
+ wed_w32(dev, MTK_WED_ADDR_ELEM_CFG0,
|
||||
+ dev->wlan.ind_cmd.particular_se_phys);
|
||||
+
|
||||
+ for (i = 0; i < dev->wlan.ind_cmd.se_group_nums; i++) {
|
||||
+ wed_w32(dev, MTK_WED_RADDR_ELEM_TBL_WDATA,
|
||||
+ dev->wlan.ind_cmd.addr_elem_phys[i] >> 4);
|
||||
+ wed_w32(dev, MTK_WED_ADDR_ELEM_TBL_CFG,
|
||||
+ MTK_WED_ADDR_ELEM_TBL_WR | (i & 0x7f));
|
||||
+
|
||||
+ val = wed_r32(dev, MTK_WED_ADDR_ELEM_TBL_CFG);
|
||||
+ while (!(val & MTK_WED_ADDR_ELEM_TBL_WR_RDY) && count++ < 100)
|
||||
+ val = wed_r32(dev, MTK_WED_ADDR_ELEM_TBL_CFG);
|
||||
+ if (count >= 100)
|
||||
+ dev_err(dev->hw->dev,
|
||||
+ "write ba session base failed\n");
|
||||
+ }
|
||||
+
|
||||
+ /* pn check init */
|
||||
+ for (i = 0; i < dev->wlan.ind_cmd.particular_sid; i++) {
|
||||
+ wed_w32(dev, MTK_WED_PN_CHECK_WDATA_M,
|
||||
+ MTK_WED_PN_CHECK_IS_FIRST);
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_PN_CHECK_CFG, MTK_WED_PN_CHECK_WR |
|
||||
+ FIELD_PREP(MTK_WED_PN_CHECK_SE_ID, i));
|
||||
+
|
||||
+ count = 0;
|
||||
+ val = wed_r32(dev, MTK_WED_PN_CHECK_CFG);
|
||||
+ while (!(val & MTK_WED_PN_CHECK_WR_RDY) && count++ < 100)
|
||||
+ val = wed_r32(dev, MTK_WED_PN_CHECK_CFG);
|
||||
+ if (count >= 100)
|
||||
+ dev_err(dev->hw->dev,
|
||||
+ "session(%d) initialization failed\n", i);
|
||||
+ }
|
||||
+
|
||||
+ wed_w32(dev, MTK_WED_RX_IND_CMD_CNT0, MTK_WED_RX_IND_CMD_DBG_CNT_EN);
|
||||
+ wed_set(dev, MTK_WED_CTRL, MTK_WED_CTRL_WED_RX_IND_CMD_EN);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
mtk_wed_start(struct mtk_wed_device *dev, u32 irq_mask)
|
||||
{
|
||||
int i;
|
||||
@@ -2216,6 +2507,10 @@ void mtk_wed_add_hw(struct device_node *
|
||||
.detach = mtk_wed_detach,
|
||||
.ppe_check = mtk_wed_ppe_check,
|
||||
.setup_tc = mtk_wed_setup_tc,
|
||||
+ .start_hw_rro = mtk_wed_start_hw_rro,
|
||||
+ .rro_rx_ring_setup = mtk_wed_rro_rx_ring_setup,
|
||||
+ .msdu_pg_rx_ring_setup = mtk_wed_msdu_pg_rx_ring_setup,
|
||||
+ .ind_rx_ring_setup = mtk_wed_ind_rx_ring_setup,
|
||||
};
|
||||
struct device_node *eth_np = eth->dev->of_node;
|
||||
struct platform_device *pdev;
|
||||
--- a/include/linux/soc/mediatek/mtk_wed.h
|
||||
+++ b/include/linux/soc/mediatek/mtk_wed.h
|
||||
@@ -10,6 +10,7 @@
|
||||
|
||||
#define MTK_WED_TX_QUEUES 2
|
||||
#define MTK_WED_RX_QUEUES 2
|
||||
+#define MTK_WED_RX_PAGE_QUEUES 3
|
||||
|
||||
#define WED_WO_STA_REC 0x6
|
||||
|
||||
@@ -99,6 +100,9 @@ struct mtk_wed_device {
|
||||
struct mtk_wed_ring txfree_ring;
|
||||
struct mtk_wed_ring tx_wdma[MTK_WED_TX_QUEUES];
|
||||
struct mtk_wed_ring rx_wdma[MTK_WED_RX_QUEUES];
|
||||
+ struct mtk_wed_ring rx_rro_ring[MTK_WED_RX_QUEUES];
|
||||
+ struct mtk_wed_ring rx_page_ring[MTK_WED_RX_PAGE_QUEUES];
|
||||
+ struct mtk_wed_ring ind_cmd_ring;
|
||||
|
||||
struct {
|
||||
int size;
|
||||
@@ -119,6 +123,13 @@ struct mtk_wed_device {
|
||||
dma_addr_t fdbk_phys;
|
||||
} rro;
|
||||
|
||||
+ struct {
|
||||
+ int size;
|
||||
+ struct mtk_wed_buf *pages;
|
||||
+ struct mtk_wed_bm_desc *desc;
|
||||
+ dma_addr_t desc_phys;
|
||||
+ } hw_rro;
|
||||
+
|
||||
/* filled by driver: */
|
||||
struct {
|
||||
union {
|
||||
@@ -137,6 +148,8 @@ struct mtk_wed_device {
|
||||
u32 wpdma_txfree;
|
||||
u32 wpdma_rx_glo;
|
||||
u32 wpdma_rx;
|
||||
+ u32 wpdma_rx_rro[MTK_WED_RX_QUEUES];
|
||||
+ u32 wpdma_rx_pg;
|
||||
|
||||
bool wcid_512;
|
||||
bool hw_rro;
|
||||
@@ -151,9 +164,20 @@ struct mtk_wed_device {
|
||||
|
||||
u8 tx_tbit[MTK_WED_TX_QUEUES];
|
||||
u8 rx_tbit[MTK_WED_RX_QUEUES];
|
||||
+ u8 rro_rx_tbit[MTK_WED_RX_QUEUES];
|
||||
+ u8 rx_pg_tbit[MTK_WED_RX_PAGE_QUEUES];
|
||||
u8 txfree_tbit;
|
||||
u8 amsdu_max_subframes;
|
||||
|
||||
+ struct {
|
||||
+ u8 se_group_nums;
|
||||
+ u16 win_size;
|
||||
+ u16 particular_sid;
|
||||
+ u32 ack_sn_addr;
|
||||
+ dma_addr_t particular_se_phys;
|
||||
+ dma_addr_t addr_elem_phys[1024];
|
||||
+ } ind_cmd;
|
||||
+
|
||||
u32 (*init_buf)(void *ptr, dma_addr_t phys, int token_id);
|
||||
int (*offload_enable)(struct mtk_wed_device *wed);
|
||||
void (*offload_disable)(struct mtk_wed_device *wed);
|
||||
@@ -192,6 +216,14 @@ struct mtk_wed_ops {
|
||||
void (*irq_set_mask)(struct mtk_wed_device *dev, u32 mask);
|
||||
int (*setup_tc)(struct mtk_wed_device *wed, struct net_device *dev,
|
||||
enum tc_setup_type type, void *type_data);
|
||||
+ void (*start_hw_rro)(struct mtk_wed_device *dev, u32 irq_mask,
|
||||
+ bool reset);
|
||||
+ void (*rro_rx_ring_setup)(struct mtk_wed_device *dev, int ring,
|
||||
+ void __iomem *regs);
|
||||
+ void (*msdu_pg_rx_ring_setup)(struct mtk_wed_device *dev, int ring,
|
||||
+ void __iomem *regs);
|
||||
+ int (*ind_rx_ring_setup)(struct mtk_wed_device *dev,
|
||||
+ void __iomem *regs);
|
||||
};
|
||||
|
||||
extern const struct mtk_wed_ops __rcu *mtk_soc_wed_ops;
|
||||
@@ -263,6 +295,15 @@ static inline bool mtk_wed_is_amsdu_supp
|
||||
#define mtk_wed_device_dma_reset(_dev) (_dev)->ops->reset_dma(_dev)
|
||||
#define mtk_wed_device_setup_tc(_dev, _netdev, _type, _type_data) \
|
||||
(_dev)->ops->setup_tc(_dev, _netdev, _type, _type_data)
|
||||
+#define mtk_wed_device_start_hw_rro(_dev, _mask, _reset) \
|
||||
+ (_dev)->ops->start_hw_rro(_dev, _mask, _reset)
|
||||
+#define mtk_wed_device_rro_rx_ring_setup(_dev, _ring, _regs) \
|
||||
+ (_dev)->ops->rro_rx_ring_setup(_dev, _ring, _regs)
|
||||
+#define mtk_wed_device_msdu_pg_rx_ring_setup(_dev, _ring, _regs) \
|
||||
+ (_dev)->ops->msdu_pg_rx_ring_setup(_dev, _ring, _regs)
|
||||
+#define mtk_wed_device_ind_rx_ring_setup(_dev, _regs) \
|
||||
+ (_dev)->ops->ind_rx_ring_setup(_dev, _regs)
|
||||
+
|
||||
#else
|
||||
static inline bool mtk_wed_device_active(struct mtk_wed_device *dev)
|
||||
{
|
||||
@@ -282,6 +323,10 @@ static inline bool mtk_wed_device_active
|
||||
#define mtk_wed_device_stop(_dev) do {} while (0)
|
||||
#define mtk_wed_device_dma_reset(_dev) do {} while (0)
|
||||
#define mtk_wed_device_setup_tc(_dev, _netdev, _type, _type_data) -EOPNOTSUPP
|
||||
+#define mtk_wed_device_start_hw_rro(_dev, _mask, _reset) do {} while (0)
|
||||
+#define mtk_wed_device_rro_rx_ring_setup(_dev, _ring, _regs) -ENODEV
|
||||
+#define mtk_wed_device_msdu_pg_rx_ring_setup(_dev, _ring, _regs) -ENODEV
|
||||
+#define mtk_wed_device_ind_rx_ring_setup(_dev, _regs) -ENODEV
|
||||
#endif
|
||||
|
||||
#endif
|
@ -0,0 +1,78 @@
|
||||
From: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 12:29:17 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: debugfs: move wed_v2 specific regs
|
||||
out of regs array
|
||||
|
||||
Move specific WED2.0 debugfs entries out of regs array. This is a
|
||||
preliminary patch to introduce WED 3.0 debugfs info.
|
||||
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_debugfs.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_debugfs.c
|
||||
@@ -151,7 +151,7 @@ DEFINE_SHOW_ATTRIBUTE(wed_txinfo);
|
||||
static int
|
||||
wed_rxinfo_show(struct seq_file *s, void *data)
|
||||
{
|
||||
- static const struct reg_dump regs[] = {
|
||||
+ static const struct reg_dump regs_common[] = {
|
||||
DUMP_STR("WPDMA RX"),
|
||||
DUMP_WPDMA_RX_RING(0),
|
||||
DUMP_WPDMA_RX_RING(1),
|
||||
@@ -169,7 +169,7 @@ wed_rxinfo_show(struct seq_file *s, void
|
||||
DUMP_WED_RING(WED_RING_RX_DATA(0)),
|
||||
DUMP_WED_RING(WED_RING_RX_DATA(1)),
|
||||
|
||||
- DUMP_STR("WED RRO"),
|
||||
+ DUMP_STR("WED WO RRO"),
|
||||
DUMP_WED_RRO_RING(WED_RROQM_MIOD_CTRL0),
|
||||
DUMP_WED(WED_RROQM_MID_MIB),
|
||||
DUMP_WED(WED_RROQM_MOD_MIB),
|
||||
@@ -180,17 +180,6 @@ wed_rxinfo_show(struct seq_file *s, void
|
||||
DUMP_WED(WED_RROQM_FDBK_ANC_MIB),
|
||||
DUMP_WED(WED_RROQM_FDBK_ANC2H_MIB),
|
||||
|
||||
- DUMP_STR("WED Route QM"),
|
||||
- DUMP_WED(WED_RTQM_R2H_MIB(0)),
|
||||
- DUMP_WED(WED_RTQM_R2Q_MIB(0)),
|
||||
- DUMP_WED(WED_RTQM_Q2H_MIB(0)),
|
||||
- DUMP_WED(WED_RTQM_R2H_MIB(1)),
|
||||
- DUMP_WED(WED_RTQM_R2Q_MIB(1)),
|
||||
- DUMP_WED(WED_RTQM_Q2H_MIB(1)),
|
||||
- DUMP_WED(WED_RTQM_Q2N_MIB),
|
||||
- DUMP_WED(WED_RTQM_Q2B_MIB),
|
||||
- DUMP_WED(WED_RTQM_PFDBK_MIB),
|
||||
-
|
||||
DUMP_STR("WED WDMA TX"),
|
||||
DUMP_WED(WED_WDMA_TX_MIB),
|
||||
DUMP_WED_RING(WED_WDMA_RING_TX),
|
||||
@@ -211,11 +200,25 @@ wed_rxinfo_show(struct seq_file *s, void
|
||||
DUMP_WED(WED_RX_BM_INTF),
|
||||
DUMP_WED(WED_RX_BM_ERR_STS),
|
||||
};
|
||||
+ static const struct reg_dump regs_wed_v2[] = {
|
||||
+ DUMP_STR("WED Route QM"),
|
||||
+ DUMP_WED(WED_RTQM_R2H_MIB(0)),
|
||||
+ DUMP_WED(WED_RTQM_R2Q_MIB(0)),
|
||||
+ DUMP_WED(WED_RTQM_Q2H_MIB(0)),
|
||||
+ DUMP_WED(WED_RTQM_R2H_MIB(1)),
|
||||
+ DUMP_WED(WED_RTQM_R2Q_MIB(1)),
|
||||
+ DUMP_WED(WED_RTQM_Q2H_MIB(1)),
|
||||
+ DUMP_WED(WED_RTQM_Q2N_MIB),
|
||||
+ DUMP_WED(WED_RTQM_Q2B_MIB),
|
||||
+ DUMP_WED(WED_RTQM_PFDBK_MIB),
|
||||
+ };
|
||||
struct mtk_wed_hw *hw = s->private;
|
||||
struct mtk_wed_device *dev = hw->wed_dev;
|
||||
|
||||
- if (dev)
|
||||
- dump_wed_regs(s, dev, regs, ARRAY_SIZE(regs));
|
||||
+ if (dev) {
|
||||
+ dump_wed_regs(s, dev, regs_common, ARRAY_SIZE(regs_common));
|
||||
+ dump_wed_regs(s, dev, regs_wed_v2, ARRAY_SIZE(regs_wed_v2));
|
||||
+ }
|
||||
|
||||
return 0;
|
||||
}
|
@ -0,0 +1,432 @@
|
||||
From: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Date: Mon, 18 Sep 2023 12:29:18 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: debugfs: add WED 3.0 debugfs entries
|
||||
|
||||
Introduce WED3.0 debugfs entries useful for debugging.
|
||||
|
||||
Co-developed-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_debugfs.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_debugfs.c
|
||||
@@ -11,6 +11,7 @@ struct reg_dump {
|
||||
u16 offset;
|
||||
u8 type;
|
||||
u8 base;
|
||||
+ u32 mask;
|
||||
};
|
||||
|
||||
enum {
|
||||
@@ -25,6 +26,8 @@ enum {
|
||||
|
||||
#define DUMP_STR(_str) { _str, 0, DUMP_TYPE_STRING }
|
||||
#define DUMP_REG(_reg, ...) { #_reg, MTK_##_reg, __VA_ARGS__ }
|
||||
+#define DUMP_REG_MASK(_reg, _mask) \
|
||||
+ { #_mask, MTK_##_reg, DUMP_TYPE_WED, 0, MTK_##_mask }
|
||||
#define DUMP_RING(_prefix, _base, ...) \
|
||||
{ _prefix " BASE", _base, __VA_ARGS__ }, \
|
||||
{ _prefix " CNT", _base + 0x4, __VA_ARGS__ }, \
|
||||
@@ -32,6 +35,7 @@ enum {
|
||||
{ _prefix " DIDX", _base + 0xc, __VA_ARGS__ }
|
||||
|
||||
#define DUMP_WED(_reg) DUMP_REG(_reg, DUMP_TYPE_WED)
|
||||
+#define DUMP_WED_MASK(_reg, _mask) DUMP_REG_MASK(_reg, _mask)
|
||||
#define DUMP_WED_RING(_base) DUMP_RING(#_base, MTK_##_base, DUMP_TYPE_WED)
|
||||
|
||||
#define DUMP_WDMA(_reg) DUMP_REG(_reg, DUMP_TYPE_WDMA)
|
||||
@@ -212,12 +216,58 @@ wed_rxinfo_show(struct seq_file *s, void
|
||||
DUMP_WED(WED_RTQM_Q2B_MIB),
|
||||
DUMP_WED(WED_RTQM_PFDBK_MIB),
|
||||
};
|
||||
+ static const struct reg_dump regs_wed_v3[] = {
|
||||
+ DUMP_STR("WED RX RRO DATA"),
|
||||
+ DUMP_WED_RING(WED_RRO_RX_D_RX(0)),
|
||||
+ DUMP_WED_RING(WED_RRO_RX_D_RX(1)),
|
||||
+
|
||||
+ DUMP_STR("WED RX MSDU PAGE"),
|
||||
+ DUMP_WED_RING(WED_RRO_MSDU_PG_CTRL0(0)),
|
||||
+ DUMP_WED_RING(WED_RRO_MSDU_PG_CTRL0(1)),
|
||||
+ DUMP_WED_RING(WED_RRO_MSDU_PG_CTRL0(2)),
|
||||
+
|
||||
+ DUMP_STR("WED RX IND CMD"),
|
||||
+ DUMP_WED(WED_IND_CMD_RX_CTRL1),
|
||||
+ DUMP_WED_MASK(WED_IND_CMD_RX_CTRL2, WED_IND_CMD_MAX_CNT),
|
||||
+ DUMP_WED_MASK(WED_IND_CMD_RX_CTRL0, WED_IND_CMD_PROC_IDX),
|
||||
+ DUMP_WED_MASK(RRO_IND_CMD_SIGNATURE, RRO_IND_CMD_DMA_IDX),
|
||||
+ DUMP_WED_MASK(WED_IND_CMD_RX_CTRL0, WED_IND_CMD_MAGIC_CNT),
|
||||
+ DUMP_WED_MASK(RRO_IND_CMD_SIGNATURE, RRO_IND_CMD_MAGIC_CNT),
|
||||
+ DUMP_WED_MASK(WED_IND_CMD_RX_CTRL0,
|
||||
+ WED_IND_CMD_PREFETCH_FREE_CNT),
|
||||
+ DUMP_WED_MASK(WED_RRO_CFG1, WED_RRO_CFG1_PARTICL_SE_ID),
|
||||
+
|
||||
+ DUMP_STR("WED ADDR ELEM"),
|
||||
+ DUMP_WED(WED_ADDR_ELEM_CFG0),
|
||||
+ DUMP_WED_MASK(WED_ADDR_ELEM_CFG1,
|
||||
+ WED_ADDR_ELEM_PREFETCH_FREE_CNT),
|
||||
+
|
||||
+ DUMP_STR("WED Route QM"),
|
||||
+ DUMP_WED(WED_RTQM_ENQ_I2Q_DMAD_CNT),
|
||||
+ DUMP_WED(WED_RTQM_ENQ_I2N_DMAD_CNT),
|
||||
+ DUMP_WED(WED_RTQM_ENQ_I2Q_PKT_CNT),
|
||||
+ DUMP_WED(WED_RTQM_ENQ_I2N_PKT_CNT),
|
||||
+ DUMP_WED(WED_RTQM_ENQ_USED_ENTRY_CNT),
|
||||
+ DUMP_WED(WED_RTQM_ENQ_ERR_CNT),
|
||||
+
|
||||
+ DUMP_WED(WED_RTQM_DEQ_DMAD_CNT),
|
||||
+ DUMP_WED(WED_RTQM_DEQ_Q2I_DMAD_CNT),
|
||||
+ DUMP_WED(WED_RTQM_DEQ_PKT_CNT),
|
||||
+ DUMP_WED(WED_RTQM_DEQ_Q2I_PKT_CNT),
|
||||
+ DUMP_WED(WED_RTQM_DEQ_USED_PFDBK_CNT),
|
||||
+ DUMP_WED(WED_RTQM_DEQ_ERR_CNT),
|
||||
+ };
|
||||
struct mtk_wed_hw *hw = s->private;
|
||||
struct mtk_wed_device *dev = hw->wed_dev;
|
||||
|
||||
if (dev) {
|
||||
dump_wed_regs(s, dev, regs_common, ARRAY_SIZE(regs_common));
|
||||
- dump_wed_regs(s, dev, regs_wed_v2, ARRAY_SIZE(regs_wed_v2));
|
||||
+ if (mtk_wed_is_v2(hw))
|
||||
+ dump_wed_regs(s, dev,
|
||||
+ regs_wed_v2, ARRAY_SIZE(regs_wed_v2));
|
||||
+ else
|
||||
+ dump_wed_regs(s, dev,
|
||||
+ regs_wed_v3, ARRAY_SIZE(regs_wed_v3));
|
||||
}
|
||||
|
||||
return 0;
|
||||
@@ -225,6 +275,314 @@ wed_rxinfo_show(struct seq_file *s, void
|
||||
DEFINE_SHOW_ATTRIBUTE(wed_rxinfo);
|
||||
|
||||
static int
|
||||
+wed_amsdu_show(struct seq_file *s, void *data)
|
||||
+{
|
||||
+ static const struct reg_dump regs[] = {
|
||||
+ DUMP_STR("WED AMDSU INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_FIFO_DMAD),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG0 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(0)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(0)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(0)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(0)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(0)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(0),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(0),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(0),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(0),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(0),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG1 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(1)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(1)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(1)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(1)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(1)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(1),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(1),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(1),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(2),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(2),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG2 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(2)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(2)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(2)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(2)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(2)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(2),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(2),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(2),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(2),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(2),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG3 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(3)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(3)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(3)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(3)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(3)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(3),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(3),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(3),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(3),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(3),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG4 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(4)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(4)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(4)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(4)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(4)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(4),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(4),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(4),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(4),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(4),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG5 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(5)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(5)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(5)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(5)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(5)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(5),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(5),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(5),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(5),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(5),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG6 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(6)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(6)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(6)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(6)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(6)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(6),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(6),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(6),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(6),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(6),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG7 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(7)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(7)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(7)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(7)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(7)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(7),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(7),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(7),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(7),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(4),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED AMDSU ENG8 INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_DMAD(8)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QFPL(8)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENI(8)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_QENO(8)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_ENG_MERG(8)),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(8),
|
||||
+ WED_AMSDU_ENG_MAX_PL_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT8(8),
|
||||
+ WED_AMSDU_ENG_MAX_QGPP_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(8),
|
||||
+ WED_AMSDU_ENG_CUR_ENTRY),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(8),
|
||||
+ WED_AMSDU_ENG_MAX_BUF_MERGED),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_ENG_CNT9(8),
|
||||
+ WED_AMSDU_ENG_MAX_MSDU_MERGED),
|
||||
+
|
||||
+ DUMP_STR("WED QMEM INFO"),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(0), WED_AMSDU_QMEM_FQ_CNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(0), WED_AMSDU_QMEM_SP_QCNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(1), WED_AMSDU_QMEM_TID0_QCNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(1), WED_AMSDU_QMEM_TID1_QCNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(2), WED_AMSDU_QMEM_TID2_QCNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(2), WED_AMSDU_QMEM_TID3_QCNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(3), WED_AMSDU_QMEM_TID4_QCNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(3), WED_AMSDU_QMEM_TID5_QCNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(4), WED_AMSDU_QMEM_TID6_QCNT),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_CNT(4), WED_AMSDU_QMEM_TID7_QCNT),
|
||||
+
|
||||
+ DUMP_STR("WED QMEM HEAD INFO"),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(0), WED_AMSDU_QMEM_FQ_HEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(0), WED_AMSDU_QMEM_SP_QHEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(1), WED_AMSDU_QMEM_TID0_QHEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(1), WED_AMSDU_QMEM_TID1_QHEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(2), WED_AMSDU_QMEM_TID2_QHEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(2), WED_AMSDU_QMEM_TID3_QHEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(3), WED_AMSDU_QMEM_TID4_QHEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(3), WED_AMSDU_QMEM_TID5_QHEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(4), WED_AMSDU_QMEM_TID6_QHEAD),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(4), WED_AMSDU_QMEM_TID7_QHEAD),
|
||||
+
|
||||
+ DUMP_STR("WED QMEM TAIL INFO"),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(5), WED_AMSDU_QMEM_FQ_TAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(5), WED_AMSDU_QMEM_SP_QTAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(6), WED_AMSDU_QMEM_TID0_QTAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(6), WED_AMSDU_QMEM_TID1_QTAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(7), WED_AMSDU_QMEM_TID2_QTAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(7), WED_AMSDU_QMEM_TID3_QTAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(8), WED_AMSDU_QMEM_TID4_QTAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(8), WED_AMSDU_QMEM_TID5_QTAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(9), WED_AMSDU_QMEM_TID6_QTAIL),
|
||||
+ DUMP_WED_MASK(WED_MON_AMSDU_QMEM_PTR(9), WED_AMSDU_QMEM_TID7_QTAIL),
|
||||
+
|
||||
+ DUMP_STR("WED HIFTXD MSDU INFO"),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(1)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(2)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(3)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(4)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(5)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(6)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(7)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(8)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(9)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(10)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(11)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(12)),
|
||||
+ DUMP_WED(WED_MON_AMSDU_HIFTXD_FETCH_MSDU(13)),
|
||||
+ };
|
||||
+ struct mtk_wed_hw *hw = s->private;
|
||||
+ struct mtk_wed_device *dev = hw->wed_dev;
|
||||
+
|
||||
+ if (dev)
|
||||
+ dump_wed_regs(s, dev, regs, ARRAY_SIZE(regs));
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+DEFINE_SHOW_ATTRIBUTE(wed_amsdu);
|
||||
+
|
||||
+static int
|
||||
+wed_rtqm_show(struct seq_file *s, void *data)
|
||||
+{
|
||||
+ static const struct reg_dump regs[] = {
|
||||
+ DUMP_STR("WED Route QM IGRS0(N2H + Recycle)"),
|
||||
+ DUMP_WED(WED_RTQM_IGRS0_I2HW_DMAD_CNT),
|
||||
+ DUMP_WED(WED_RTQM_IGRS0_I2H_DMAD_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS0_I2H_DMAD_CNT(1)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS0_I2HW_PKT_CNT),
|
||||
+ DUMP_WED(WED_RTQM_IGRS0_I2H_PKT_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS0_I2H_PKT_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS0_FDROP_CNT),
|
||||
+
|
||||
+ DUMP_STR("WED Route QM IGRS1(Legacy)"),
|
||||
+ DUMP_WED(WED_RTQM_IGRS1_I2HW_DMAD_CNT),
|
||||
+ DUMP_WED(WED_RTQM_IGRS1_I2H_DMAD_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS1_I2H_DMAD_CNT(1)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS1_I2HW_PKT_CNT),
|
||||
+ DUMP_WED(WED_RTQM_IGRS1_I2H_PKT_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS1_I2H_PKT_CNT(1)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS1_FDROP_CNT),
|
||||
+
|
||||
+ DUMP_STR("WED Route QM IGRS2(RRO3.0)"),
|
||||
+ DUMP_WED(WED_RTQM_IGRS2_I2HW_DMAD_CNT),
|
||||
+ DUMP_WED(WED_RTQM_IGRS2_I2H_DMAD_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS2_I2H_DMAD_CNT(1)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS2_I2HW_PKT_CNT),
|
||||
+ DUMP_WED(WED_RTQM_IGRS2_I2H_PKT_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS2_I2H_PKT_CNT(1)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS2_FDROP_CNT),
|
||||
+
|
||||
+ DUMP_STR("WED Route QM IGRS3(DEBUG)"),
|
||||
+ DUMP_WED(WED_RTQM_IGRS2_I2HW_DMAD_CNT),
|
||||
+ DUMP_WED(WED_RTQM_IGRS3_I2H_DMAD_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS3_I2H_DMAD_CNT(1)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS3_I2HW_PKT_CNT),
|
||||
+ DUMP_WED(WED_RTQM_IGRS3_I2H_PKT_CNT(0)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS3_I2H_PKT_CNT(1)),
|
||||
+ DUMP_WED(WED_RTQM_IGRS3_FDROP_CNT),
|
||||
+ };
|
||||
+ struct mtk_wed_hw *hw = s->private;
|
||||
+ struct mtk_wed_device *dev = hw->wed_dev;
|
||||
+
|
||||
+ if (dev)
|
||||
+ dump_wed_regs(s, dev, regs, ARRAY_SIZE(regs));
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+DEFINE_SHOW_ATTRIBUTE(wed_rtqm);
|
||||
+
|
||||
+static int
|
||||
+wed_rro_show(struct seq_file *s, void *data)
|
||||
+{
|
||||
+ static const struct reg_dump regs[] = {
|
||||
+ DUMP_STR("RRO/IND CMD CNT"),
|
||||
+ DUMP_WED(WED_RX_IND_CMD_CNT(1)),
|
||||
+ DUMP_WED(WED_RX_IND_CMD_CNT(2)),
|
||||
+ DUMP_WED(WED_RX_IND_CMD_CNT(3)),
|
||||
+ DUMP_WED(WED_RX_IND_CMD_CNT(4)),
|
||||
+ DUMP_WED(WED_RX_IND_CMD_CNT(5)),
|
||||
+ DUMP_WED(WED_RX_IND_CMD_CNT(6)),
|
||||
+ DUMP_WED(WED_RX_IND_CMD_CNT(7)),
|
||||
+ DUMP_WED(WED_RX_IND_CMD_CNT(8)),
|
||||
+ DUMP_WED_MASK(WED_RX_IND_CMD_CNT(9),
|
||||
+ WED_IND_CMD_MAGIC_CNT_FAIL_CNT),
|
||||
+
|
||||
+ DUMP_WED(WED_RX_ADDR_ELEM_CNT(0)),
|
||||
+ DUMP_WED_MASK(WED_RX_ADDR_ELEM_CNT(1),
|
||||
+ WED_ADDR_ELEM_SIG_FAIL_CNT),
|
||||
+ DUMP_WED(WED_RX_MSDU_PG_CNT(1)),
|
||||
+ DUMP_WED(WED_RX_MSDU_PG_CNT(2)),
|
||||
+ DUMP_WED(WED_RX_MSDU_PG_CNT(3)),
|
||||
+ DUMP_WED(WED_RX_MSDU_PG_CNT(4)),
|
||||
+ DUMP_WED(WED_RX_MSDU_PG_CNT(5)),
|
||||
+ DUMP_WED_MASK(WED_RX_PN_CHK_CNT,
|
||||
+ WED_PN_CHK_FAIL_CNT),
|
||||
+ };
|
||||
+ struct mtk_wed_hw *hw = s->private;
|
||||
+ struct mtk_wed_device *dev = hw->wed_dev;
|
||||
+
|
||||
+ if (dev)
|
||||
+ dump_wed_regs(s, dev, regs, ARRAY_SIZE(regs));
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+DEFINE_SHOW_ATTRIBUTE(wed_rro);
|
||||
+
|
||||
+static int
|
||||
mtk_wed_reg_set(void *data, u64 val)
|
||||
{
|
||||
struct mtk_wed_hw *hw = data;
|
||||
@@ -264,7 +622,16 @@ void mtk_wed_hw_add_debugfs(struct mtk_w
|
||||
debugfs_create_u32("regidx", 0600, dir, &hw->debugfs_reg);
|
||||
debugfs_create_file_unsafe("regval", 0600, dir, hw, &fops_regval);
|
||||
debugfs_create_file_unsafe("txinfo", 0400, dir, hw, &wed_txinfo_fops);
|
||||
- if (!mtk_wed_is_v1(hw))
|
||||
+ if (!mtk_wed_is_v1(hw)) {
|
||||
debugfs_create_file_unsafe("rxinfo", 0400, dir, hw,
|
||||
&wed_rxinfo_fops);
|
||||
+ if (mtk_wed_is_v3_or_greater(hw)) {
|
||||
+ debugfs_create_file_unsafe("amsdu", 0400, dir, hw,
|
||||
+ &wed_amsdu_fops);
|
||||
+ debugfs_create_file_unsafe("rtqm", 0400, dir, hw,
|
||||
+ &wed_rtqm_fops);
|
||||
+ debugfs_create_file_unsafe("rro", 0400, dir, hw,
|
||||
+ &wed_rro_fops);
|
||||
+ }
|
||||
+ }
|
||||
}
|
@ -0,0 +1,587 @@
|
||||
From: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Date: Mon, 18 Sep 2023 12:29:19 +0200
|
||||
Subject: [PATCH] net: ethernet: mtk_wed: add wed 3.0 reset support
|
||||
|
||||
Introduce support for resetting Wireless Ethernet Dispatcher 3.0
|
||||
available on MT988 SoC.
|
||||
|
||||
Co-developed-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
Signed-off-by: Sujuan Chen <sujuan.chen@mediatek.com>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed.c
|
||||
@@ -149,6 +149,90 @@ mtk_wdma_read_reset(struct mtk_wed_devic
|
||||
return wdma_r32(dev, MTK_WDMA_GLO_CFG);
|
||||
}
|
||||
|
||||
+static void
|
||||
+mtk_wdma_v3_rx_reset(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+ u32 status;
|
||||
+
|
||||
+ if (!mtk_wed_is_v3_or_greater(dev->hw))
|
||||
+ return;
|
||||
+
|
||||
+ wdma_clr(dev, MTK_WDMA_PREF_TX_CFG, MTK_WDMA_PREF_TX_CFG_PREF_EN);
|
||||
+ wdma_clr(dev, MTK_WDMA_PREF_RX_CFG, MTK_WDMA_PREF_RX_CFG_PREF_EN);
|
||||
+
|
||||
+ if (read_poll_timeout(wdma_r32, status,
|
||||
+ !(status & MTK_WDMA_PREF_TX_CFG_PREF_BUSY),
|
||||
+ 0, 10000, false, dev, MTK_WDMA_PREF_TX_CFG))
|
||||
+ dev_err(dev->hw->dev, "rx reset failed\n");
|
||||
+
|
||||
+ if (read_poll_timeout(wdma_r32, status,
|
||||
+ !(status & MTK_WDMA_PREF_RX_CFG_PREF_BUSY),
|
||||
+ 0, 10000, false, dev, MTK_WDMA_PREF_RX_CFG))
|
||||
+ dev_err(dev->hw->dev, "rx reset failed\n");
|
||||
+
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_TX_CFG, MTK_WDMA_WRBK_TX_CFG_WRBK_EN);
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_RX_CFG, MTK_WDMA_WRBK_RX_CFG_WRBK_EN);
|
||||
+
|
||||
+ if (read_poll_timeout(wdma_r32, status,
|
||||
+ !(status & MTK_WDMA_WRBK_TX_CFG_WRBK_BUSY),
|
||||
+ 0, 10000, false, dev, MTK_WDMA_WRBK_TX_CFG))
|
||||
+ dev_err(dev->hw->dev, "rx reset failed\n");
|
||||
+
|
||||
+ if (read_poll_timeout(wdma_r32, status,
|
||||
+ !(status & MTK_WDMA_WRBK_RX_CFG_WRBK_BUSY),
|
||||
+ 0, 10000, false, dev, MTK_WDMA_WRBK_RX_CFG))
|
||||
+ dev_err(dev->hw->dev, "rx reset failed\n");
|
||||
+
|
||||
+ /* prefetch FIFO */
|
||||
+ wdma_w32(dev, MTK_WDMA_PREF_RX_FIFO_CFG,
|
||||
+ MTK_WDMA_PREF_RX_FIFO_CFG_RING0_CLEAR |
|
||||
+ MTK_WDMA_PREF_RX_FIFO_CFG_RING1_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_PREF_RX_FIFO_CFG,
|
||||
+ MTK_WDMA_PREF_RX_FIFO_CFG_RING0_CLEAR |
|
||||
+ MTK_WDMA_PREF_RX_FIFO_CFG_RING1_CLEAR);
|
||||
+
|
||||
+ /* core FIFO */
|
||||
+ wdma_w32(dev, MTK_WDMA_XDMA_RX_FIFO_CFG,
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_PAR_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_CMD_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_DMAD_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_ARR_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_LEN_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_WID_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_BID_FIFO_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_XDMA_RX_FIFO_CFG,
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_PAR_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_CMD_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_DMAD_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_ARR_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_LEN_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_WID_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_RX_FIFO_CFG_RX_BID_FIFO_CLEAR);
|
||||
+
|
||||
+ /* writeback FIFO */
|
||||
+ wdma_w32(dev, MTK_WDMA_WRBK_RX_FIFO_CFG(0),
|
||||
+ MTK_WDMA_WRBK_RX_FIFO_CFG_RING_CLEAR);
|
||||
+ wdma_w32(dev, MTK_WDMA_WRBK_RX_FIFO_CFG(1),
|
||||
+ MTK_WDMA_WRBK_RX_FIFO_CFG_RING_CLEAR);
|
||||
+
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_RX_FIFO_CFG(0),
|
||||
+ MTK_WDMA_WRBK_RX_FIFO_CFG_RING_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_RX_FIFO_CFG(1),
|
||||
+ MTK_WDMA_WRBK_RX_FIFO_CFG_RING_CLEAR);
|
||||
+
|
||||
+ /* prefetch ring status */
|
||||
+ wdma_w32(dev, MTK_WDMA_PREF_SIDX_CFG,
|
||||
+ MTK_WDMA_PREF_SIDX_CFG_RX_RING_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_PREF_SIDX_CFG,
|
||||
+ MTK_WDMA_PREF_SIDX_CFG_RX_RING_CLEAR);
|
||||
+
|
||||
+ /* writeback ring status */
|
||||
+ wdma_w32(dev, MTK_WDMA_WRBK_SIDX_CFG,
|
||||
+ MTK_WDMA_WRBK_SIDX_CFG_RX_RING_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_SIDX_CFG,
|
||||
+ MTK_WDMA_WRBK_SIDX_CFG_RX_RING_CLEAR);
|
||||
+}
|
||||
+
|
||||
static int
|
||||
mtk_wdma_rx_reset(struct mtk_wed_device *dev)
|
||||
{
|
||||
@@ -161,6 +245,7 @@ mtk_wdma_rx_reset(struct mtk_wed_device
|
||||
if (ret)
|
||||
dev_err(dev->hw->dev, "rx reset failed\n");
|
||||
|
||||
+ mtk_wdma_v3_rx_reset(dev);
|
||||
wdma_w32(dev, MTK_WDMA_RESET_IDX, MTK_WDMA_RESET_IDX_RX);
|
||||
wdma_w32(dev, MTK_WDMA_RESET_IDX, 0);
|
||||
|
||||
@@ -193,6 +278,84 @@ mtk_wed_poll_busy(struct mtk_wed_device
|
||||
}
|
||||
|
||||
static void
|
||||
+mtk_wdma_v3_tx_reset(struct mtk_wed_device *dev)
|
||||
+{
|
||||
+ u32 status;
|
||||
+
|
||||
+ if (!mtk_wed_is_v3_or_greater(dev->hw))
|
||||
+ return;
|
||||
+
|
||||
+ wdma_clr(dev, MTK_WDMA_PREF_TX_CFG, MTK_WDMA_PREF_TX_CFG_PREF_EN);
|
||||
+ wdma_clr(dev, MTK_WDMA_PREF_RX_CFG, MTK_WDMA_PREF_RX_CFG_PREF_EN);
|
||||
+
|
||||
+ if (read_poll_timeout(wdma_r32, status,
|
||||
+ !(status & MTK_WDMA_PREF_TX_CFG_PREF_BUSY),
|
||||
+ 0, 10000, false, dev, MTK_WDMA_PREF_TX_CFG))
|
||||
+ dev_err(dev->hw->dev, "tx reset failed\n");
|
||||
+
|
||||
+ if (read_poll_timeout(wdma_r32, status,
|
||||
+ !(status & MTK_WDMA_PREF_RX_CFG_PREF_BUSY),
|
||||
+ 0, 10000, false, dev, MTK_WDMA_PREF_RX_CFG))
|
||||
+ dev_err(dev->hw->dev, "tx reset failed\n");
|
||||
+
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_TX_CFG, MTK_WDMA_WRBK_TX_CFG_WRBK_EN);
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_RX_CFG, MTK_WDMA_WRBK_RX_CFG_WRBK_EN);
|
||||
+
|
||||
+ if (read_poll_timeout(wdma_r32, status,
|
||||
+ !(status & MTK_WDMA_WRBK_TX_CFG_WRBK_BUSY),
|
||||
+ 0, 10000, false, dev, MTK_WDMA_WRBK_TX_CFG))
|
||||
+ dev_err(dev->hw->dev, "tx reset failed\n");
|
||||
+
|
||||
+ if (read_poll_timeout(wdma_r32, status,
|
||||
+ !(status & MTK_WDMA_WRBK_RX_CFG_WRBK_BUSY),
|
||||
+ 0, 10000, false, dev, MTK_WDMA_WRBK_RX_CFG))
|
||||
+ dev_err(dev->hw->dev, "tx reset failed\n");
|
||||
+
|
||||
+ /* prefetch FIFO */
|
||||
+ wdma_w32(dev, MTK_WDMA_PREF_TX_FIFO_CFG,
|
||||
+ MTK_WDMA_PREF_TX_FIFO_CFG_RING0_CLEAR |
|
||||
+ MTK_WDMA_PREF_TX_FIFO_CFG_RING1_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_PREF_TX_FIFO_CFG,
|
||||
+ MTK_WDMA_PREF_TX_FIFO_CFG_RING0_CLEAR |
|
||||
+ MTK_WDMA_PREF_TX_FIFO_CFG_RING1_CLEAR);
|
||||
+
|
||||
+ /* core FIFO */
|
||||
+ wdma_w32(dev, MTK_WDMA_XDMA_TX_FIFO_CFG,
|
||||
+ MTK_WDMA_XDMA_TX_FIFO_CFG_TX_PAR_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_TX_FIFO_CFG_TX_CMD_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_TX_FIFO_CFG_TX_DMAD_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_TX_FIFO_CFG_TX_ARR_FIFO_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_XDMA_TX_FIFO_CFG,
|
||||
+ MTK_WDMA_XDMA_TX_FIFO_CFG_TX_PAR_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_TX_FIFO_CFG_TX_CMD_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_TX_FIFO_CFG_TX_DMAD_FIFO_CLEAR |
|
||||
+ MTK_WDMA_XDMA_TX_FIFO_CFG_TX_ARR_FIFO_CLEAR);
|
||||
+
|
||||
+ /* writeback FIFO */
|
||||
+ wdma_w32(dev, MTK_WDMA_WRBK_TX_FIFO_CFG(0),
|
||||
+ MTK_WDMA_WRBK_TX_FIFO_CFG_RING_CLEAR);
|
||||
+ wdma_w32(dev, MTK_WDMA_WRBK_TX_FIFO_CFG(1),
|
||||
+ MTK_WDMA_WRBK_TX_FIFO_CFG_RING_CLEAR);
|
||||
+
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_TX_FIFO_CFG(0),
|
||||
+ MTK_WDMA_WRBK_TX_FIFO_CFG_RING_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_TX_FIFO_CFG(1),
|
||||
+ MTK_WDMA_WRBK_TX_FIFO_CFG_RING_CLEAR);
|
||||
+
|
||||
+ /* prefetch ring status */
|
||||
+ wdma_w32(dev, MTK_WDMA_PREF_SIDX_CFG,
|
||||
+ MTK_WDMA_PREF_SIDX_CFG_TX_RING_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_PREF_SIDX_CFG,
|
||||
+ MTK_WDMA_PREF_SIDX_CFG_TX_RING_CLEAR);
|
||||
+
|
||||
+ /* writeback ring status */
|
||||
+ wdma_w32(dev, MTK_WDMA_WRBK_SIDX_CFG,
|
||||
+ MTK_WDMA_WRBK_SIDX_CFG_TX_RING_CLEAR);
|
||||
+ wdma_clr(dev, MTK_WDMA_WRBK_SIDX_CFG,
|
||||
+ MTK_WDMA_WRBK_SIDX_CFG_TX_RING_CLEAR);
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
mtk_wdma_tx_reset(struct mtk_wed_device *dev)
|
||||
{
|
||||
u32 status, mask = MTK_WDMA_GLO_CFG_TX_DMA_BUSY;
|
||||
@@ -203,6 +366,7 @@ mtk_wdma_tx_reset(struct mtk_wed_device
|
||||
!(status & mask), 0, 10000))
|
||||
dev_err(dev->hw->dev, "tx reset failed\n");
|
||||
|
||||
+ mtk_wdma_v3_tx_reset(dev);
|
||||
wdma_w32(dev, MTK_WDMA_RESET_IDX, MTK_WDMA_RESET_IDX_TX);
|
||||
wdma_w32(dev, MTK_WDMA_RESET_IDX, 0);
|
||||
|
||||
@@ -1406,13 +1570,33 @@ mtk_wed_rx_reset(struct mtk_wed_device *
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
+ if (dev->wlan.hw_rro) {
|
||||
+ wed_clr(dev, MTK_WED_CTRL, MTK_WED_CTRL_WED_RX_IND_CMD_EN);
|
||||
+ mtk_wed_poll_busy(dev, MTK_WED_RRO_RX_HW_STS,
|
||||
+ MTK_WED_RX_IND_CMD_BUSY);
|
||||
+ mtk_wed_reset(dev, MTK_WED_RESET_RRO_RX_TO_PG);
|
||||
+ }
|
||||
+
|
||||
wed_clr(dev, MTK_WED_WPDMA_RX_D_GLO_CFG, MTK_WED_WPDMA_RX_D_RX_DRV_EN);
|
||||
ret = mtk_wed_poll_busy(dev, MTK_WED_WPDMA_RX_D_GLO_CFG,
|
||||
MTK_WED_WPDMA_RX_D_RX_DRV_BUSY);
|
||||
+ if (!ret && mtk_wed_is_v3_or_greater(dev->hw))
|
||||
+ ret = mtk_wed_poll_busy(dev, MTK_WED_WPDMA_RX_D_PREF_CFG,
|
||||
+ MTK_WED_WPDMA_RX_D_PREF_BUSY);
|
||||
if (ret) {
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WPDMA_INT_AGENT);
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WPDMA_RX_D_DRV);
|
||||
} else {
|
||||
+ if (mtk_wed_is_v3_or_greater(dev->hw)) {
|
||||
+ /* 1.a. disable prefetch HW */
|
||||
+ wed_clr(dev, MTK_WED_WPDMA_RX_D_PREF_CFG,
|
||||
+ MTK_WED_WPDMA_RX_D_PREF_EN);
|
||||
+ mtk_wed_poll_busy(dev, MTK_WED_WPDMA_RX_D_PREF_CFG,
|
||||
+ MTK_WED_WPDMA_RX_D_PREF_BUSY);
|
||||
+ wed_w32(dev, MTK_WED_WPDMA_RX_D_RST_IDX,
|
||||
+ MTK_WED_WPDMA_RX_D_RST_DRV_IDX_ALL);
|
||||
+ }
|
||||
+
|
||||
wed_w32(dev, MTK_WED_WPDMA_RX_D_RST_IDX,
|
||||
MTK_WED_WPDMA_RX_D_RST_CRX_IDX |
|
||||
MTK_WED_WPDMA_RX_D_RST_DRV_IDX);
|
||||
@@ -1440,23 +1624,52 @@ mtk_wed_rx_reset(struct mtk_wed_device *
|
||||
wed_w32(dev, MTK_WED_RROQM_RST_IDX, 0);
|
||||
}
|
||||
|
||||
+ if (dev->wlan.hw_rro) {
|
||||
+ /* disable rro msdu page drv */
|
||||
+ wed_clr(dev, MTK_WED_RRO_MSDU_PG_RING2_CFG,
|
||||
+ MTK_WED_RRO_MSDU_PG_DRV_EN);
|
||||
+
|
||||
+ /* disable rro data drv */
|
||||
+ wed_clr(dev, MTK_WED_RRO_RX_D_CFG(2), MTK_WED_RRO_RX_D_DRV_EN);
|
||||
+
|
||||
+ /* rro msdu page drv reset */
|
||||
+ wed_w32(dev, MTK_WED_RRO_MSDU_PG_RING2_CFG,
|
||||
+ MTK_WED_RRO_MSDU_PG_DRV_CLR);
|
||||
+ mtk_wed_poll_busy(dev, MTK_WED_RRO_MSDU_PG_RING2_CFG,
|
||||
+ MTK_WED_RRO_MSDU_PG_DRV_CLR);
|
||||
+
|
||||
+ /* rro data drv reset */
|
||||
+ wed_w32(dev, MTK_WED_RRO_RX_D_CFG(2),
|
||||
+ MTK_WED_RRO_RX_D_DRV_CLR);
|
||||
+ mtk_wed_poll_busy(dev, MTK_WED_RRO_RX_D_CFG(2),
|
||||
+ MTK_WED_RRO_RX_D_DRV_CLR);
|
||||
+ }
|
||||
+
|
||||
/* reset route qm */
|
||||
wed_clr(dev, MTK_WED_CTRL, MTK_WED_CTRL_RX_ROUTE_QM_EN);
|
||||
ret = mtk_wed_poll_busy(dev, MTK_WED_CTRL,
|
||||
MTK_WED_CTRL_RX_ROUTE_QM_BUSY);
|
||||
- if (ret)
|
||||
+ if (ret) {
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_RX_ROUTE_QM);
|
||||
- else
|
||||
- wed_set(dev, MTK_WED_RTQM_GLO_CFG,
|
||||
- MTK_WED_RTQM_Q_RST);
|
||||
+ } else if (mtk_wed_is_v3_or_greater(dev->hw)) {
|
||||
+ wed_set(dev, MTK_WED_RTQM_RST, BIT(0));
|
||||
+ wed_clr(dev, MTK_WED_RTQM_RST, BIT(0));
|
||||
+ mtk_wed_reset(dev, MTK_WED_RESET_RX_ROUTE_QM);
|
||||
+ } else {
|
||||
+ wed_set(dev, MTK_WED_RTQM_GLO_CFG, MTK_WED_RTQM_Q_RST);
|
||||
+ }
|
||||
|
||||
/* reset tx wdma */
|
||||
mtk_wdma_tx_reset(dev);
|
||||
|
||||
/* reset tx wdma drv */
|
||||
wed_clr(dev, MTK_WED_WDMA_GLO_CFG, MTK_WED_WDMA_GLO_CFG_TX_DRV_EN);
|
||||
- mtk_wed_poll_busy(dev, MTK_WED_CTRL,
|
||||
- MTK_WED_CTRL_WDMA_INT_AGENT_BUSY);
|
||||
+ if (mtk_wed_is_v3_or_greater(dev->hw))
|
||||
+ mtk_wed_poll_busy(dev, MTK_WED_WPDMA_STATUS,
|
||||
+ MTK_WED_WPDMA_STATUS_TX_DRV);
|
||||
+ else
|
||||
+ mtk_wed_poll_busy(dev, MTK_WED_CTRL,
|
||||
+ MTK_WED_CTRL_WDMA_INT_AGENT_BUSY);
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WDMA_TX_DRV);
|
||||
|
||||
/* reset wed rx dma */
|
||||
@@ -1477,6 +1690,14 @@ mtk_wed_rx_reset(struct mtk_wed_device *
|
||||
MTK_WED_CTRL_WED_RX_BM_BUSY);
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_RX_BM);
|
||||
|
||||
+ if (dev->wlan.hw_rro) {
|
||||
+ wed_clr(dev, MTK_WED_CTRL, MTK_WED_CTRL_WED_RX_PG_BM_EN);
|
||||
+ mtk_wed_poll_busy(dev, MTK_WED_CTRL,
|
||||
+ MTK_WED_CTRL_WED_RX_PG_BM_BUSY);
|
||||
+ wed_set(dev, MTK_WED_RESET, MTK_WED_RESET_RX_PG_BM);
|
||||
+ wed_clr(dev, MTK_WED_RESET, MTK_WED_RESET_RX_PG_BM);
|
||||
+ }
|
||||
+
|
||||
/* wo change to enable state */
|
||||
val = MTK_WED_WO_STATE_ENABLE;
|
||||
ret = mtk_wed_mcu_send_msg(wo, MTK_WED_MODULE_ID_WO,
|
||||
@@ -1494,6 +1715,7 @@ mtk_wed_rx_reset(struct mtk_wed_device *
|
||||
false);
|
||||
}
|
||||
mtk_wed_free_rx_buffer(dev);
|
||||
+ mtk_wed_hwrro_free_buffer(dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -1527,15 +1749,41 @@ mtk_wed_reset_dma(struct mtk_wed_device
|
||||
|
||||
/* 2. reset WDMA rx DMA */
|
||||
busy = !!mtk_wdma_rx_reset(dev);
|
||||
- wed_clr(dev, MTK_WED_WDMA_GLO_CFG, MTK_WED_WDMA_GLO_CFG_RX_DRV_EN);
|
||||
+ if (mtk_wed_is_v3_or_greater(dev->hw)) {
|
||||
+ val = MTK_WED_WDMA_GLO_CFG_RX_DIS_FSM_AUTO_IDLE |
|
||||
+ wed_r32(dev, MTK_WED_WDMA_GLO_CFG);
|
||||
+ val &= ~MTK_WED_WDMA_GLO_CFG_RX_DRV_EN;
|
||||
+ wed_w32(dev, MTK_WED_WDMA_GLO_CFG, val);
|
||||
+ } else {
|
||||
+ wed_clr(dev, MTK_WED_WDMA_GLO_CFG,
|
||||
+ MTK_WED_WDMA_GLO_CFG_RX_DRV_EN);
|
||||
+ }
|
||||
+
|
||||
if (!busy)
|
||||
busy = mtk_wed_poll_busy(dev, MTK_WED_WDMA_GLO_CFG,
|
||||
MTK_WED_WDMA_GLO_CFG_RX_DRV_BUSY);
|
||||
+ if (!busy && mtk_wed_is_v3_or_greater(dev->hw))
|
||||
+ busy = mtk_wed_poll_busy(dev, MTK_WED_WDMA_RX_PREF_CFG,
|
||||
+ MTK_WED_WDMA_RX_PREF_BUSY);
|
||||
|
||||
if (busy) {
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WDMA_INT_AGENT);
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WDMA_RX_DRV);
|
||||
} else {
|
||||
+ if (mtk_wed_is_v3_or_greater(dev->hw)) {
|
||||
+ /* 1.a. disable prefetch HW */
|
||||
+ wed_clr(dev, MTK_WED_WDMA_RX_PREF_CFG,
|
||||
+ MTK_WED_WDMA_RX_PREF_EN);
|
||||
+ mtk_wed_poll_busy(dev, MTK_WED_WDMA_RX_PREF_CFG,
|
||||
+ MTK_WED_WDMA_RX_PREF_BUSY);
|
||||
+ wed_clr(dev, MTK_WED_WDMA_RX_PREF_CFG,
|
||||
+ MTK_WED_WDMA_RX_PREF_DDONE2_EN);
|
||||
+
|
||||
+ /* 2. Reset dma index */
|
||||
+ wed_w32(dev, MTK_WED_WDMA_RESET_IDX,
|
||||
+ MTK_WED_WDMA_RESET_IDX_RX_ALL);
|
||||
+ }
|
||||
+
|
||||
wed_w32(dev, MTK_WED_WDMA_RESET_IDX,
|
||||
MTK_WED_WDMA_RESET_IDX_RX | MTK_WED_WDMA_RESET_IDX_DRV);
|
||||
wed_w32(dev, MTK_WED_WDMA_RESET_IDX, 0);
|
||||
@@ -1551,8 +1799,13 @@ mtk_wed_reset_dma(struct mtk_wed_device
|
||||
wed_clr(dev, MTK_WED_CTRL, MTK_WED_CTRL_WED_TX_FREE_AGENT_EN);
|
||||
|
||||
for (i = 0; i < 100; i++) {
|
||||
- val = wed_r32(dev, MTK_WED_TX_BM_INTF);
|
||||
- if (FIELD_GET(MTK_WED_TX_BM_INTF_TKFIFO_FDEP, val) == 0x40)
|
||||
+ if (mtk_wed_is_v1(dev->hw))
|
||||
+ val = FIELD_GET(MTK_WED_TX_BM_INTF_TKFIFO_FDEP,
|
||||
+ wed_r32(dev, MTK_WED_TX_BM_INTF));
|
||||
+ else
|
||||
+ val = FIELD_GET(MTK_WED_TX_TKID_INTF_TKFIFO_FDEP,
|
||||
+ wed_r32(dev, MTK_WED_TX_TKID_INTF));
|
||||
+ if (val == 0x40)
|
||||
break;
|
||||
}
|
||||
|
||||
@@ -1574,6 +1827,8 @@ mtk_wed_reset_dma(struct mtk_wed_device
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WPDMA_INT_AGENT);
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WPDMA_TX_DRV);
|
||||
mtk_wed_reset(dev, MTK_WED_RESET_WPDMA_RX_DRV);
|
||||
+ if (mtk_wed_is_v3_or_greater(dev->hw))
|
||||
+ wed_w32(dev, MTK_WED_RX1_CTRL2, 0);
|
||||
} else {
|
||||
wed_w32(dev, MTK_WED_WPDMA_RESET_IDX,
|
||||
MTK_WED_WPDMA_RESET_IDX_TX |
|
||||
@@ -1590,7 +1845,14 @@ mtk_wed_reset_dma(struct mtk_wed_device
|
||||
wed_w32(dev, MTK_WED_RESET_IDX, 0);
|
||||
}
|
||||
|
||||
- mtk_wed_rx_reset(dev);
|
||||
+ if (mtk_wed_is_v3_or_greater(dev->hw)) {
|
||||
+ /* reset amsdu engine */
|
||||
+ wed_clr(dev, MTK_WED_CTRL, MTK_WED_CTRL_TX_AMSDU_EN);
|
||||
+ mtk_wed_reset(dev, MTK_WED_RESET_TX_AMSDU);
|
||||
+ }
|
||||
+
|
||||
+ if (mtk_wed_get_rx_capa(dev))
|
||||
+ mtk_wed_rx_reset(dev);
|
||||
}
|
||||
|
||||
static int
|
||||
@@ -1842,6 +2104,7 @@ mtk_wed_dma_enable(struct mtk_wed_device
|
||||
MTK_WED_WPDMA_GLO_CFG_RX_DRV_UNS_VER_FORCE_4);
|
||||
|
||||
wdma_set(dev, MTK_WDMA_PREF_RX_CFG, MTK_WDMA_PREF_RX_CFG_PREF_EN);
|
||||
+ wdma_set(dev, MTK_WDMA_WRBK_RX_CFG, MTK_WDMA_WRBK_RX_CFG_WRBK_EN);
|
||||
}
|
||||
|
||||
wed_clr(dev, MTK_WED_WPDMA_GLO_CFG,
|
||||
@@ -1905,6 +2168,12 @@ mtk_wed_start_hw_rro(struct mtk_wed_devi
|
||||
if (!mtk_wed_get_rx_capa(dev) || !dev->wlan.hw_rro)
|
||||
return;
|
||||
|
||||
+ if (reset) {
|
||||
+ wed_set(dev, MTK_WED_RRO_MSDU_PG_RING2_CFG,
|
||||
+ MTK_WED_RRO_MSDU_PG_DRV_EN);
|
||||
+ return;
|
||||
+ }
|
||||
+
|
||||
wed_set(dev, MTK_WED_RRO_RX_D_CFG(2), MTK_WED_RRO_MSDU_PG_DRV_CLR);
|
||||
wed_w32(dev, MTK_WED_RRO_MSDU_PG_RING2_CFG,
|
||||
MTK_WED_RRO_MSDU_PG_DRV_CLR);
|
||||
--- a/drivers/net/ethernet/mediatek/mtk_wed_regs.h
|
||||
+++ b/drivers/net/ethernet/mediatek/mtk_wed_regs.h
|
||||
@@ -28,6 +28,8 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_RESET 0x008
|
||||
#define MTK_WED_RESET_TX_BM BIT(0)
|
||||
#define MTK_WED_RESET_RX_BM BIT(1)
|
||||
+#define MTK_WED_RESET_RX_PG_BM BIT(2)
|
||||
+#define MTK_WED_RESET_RRO_RX_TO_PG BIT(3)
|
||||
#define MTK_WED_RESET_TX_FREE_AGENT BIT(4)
|
||||
#define MTK_WED_RESET_WPDMA_TX_DRV BIT(8)
|
||||
#define MTK_WED_RESET_WPDMA_RX_DRV BIT(9)
|
||||
@@ -106,6 +108,9 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_STATUS 0x060
|
||||
#define MTK_WED_STATUS_TX GENMASK(15, 8)
|
||||
|
||||
+#define MTK_WED_WPDMA_STATUS 0x068
|
||||
+#define MTK_WED_WPDMA_STATUS_TX_DRV GENMASK(15, 8)
|
||||
+
|
||||
#define MTK_WED_TX_BM_CTRL 0x080
|
||||
#define MTK_WED_TX_BM_CTRL_VLD_GRP_NUM GENMASK(6, 0)
|
||||
#define MTK_WED_TX_BM_CTRL_RSV_GRP_NUM GENMASK(22, 16)
|
||||
@@ -140,6 +145,9 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_TX_TKID_CTRL_RSV_GRP_NUM GENMASK(22, 16)
|
||||
#define MTK_WED_TX_TKID_CTRL_PAUSE BIT(28)
|
||||
|
||||
+#define MTK_WED_TX_TKID_INTF 0x0dc
|
||||
+#define MTK_WED_TX_TKID_INTF_TKFIFO_FDEP GENMASK(25, 16)
|
||||
+
|
||||
#define MTK_WED_TX_TKID_CTRL_VLD_GRP_NUM_V3 GENMASK(7, 0)
|
||||
#define MTK_WED_TX_TKID_CTRL_RSV_GRP_NUM_V3 GENMASK(23, 16)
|
||||
|
||||
@@ -190,6 +198,7 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_RING_RX_DATA(_n) (0x420 + (_n) * 0x10)
|
||||
|
||||
#define MTK_WED_SCR0 0x3c0
|
||||
+#define MTK_WED_RX1_CTRL2 0x418
|
||||
#define MTK_WED_WPDMA_INT_TRIGGER 0x504
|
||||
#define MTK_WED_WPDMA_INT_TRIGGER_RX_DONE BIT(1)
|
||||
#define MTK_WED_WPDMA_INT_TRIGGER_TX_DONE GENMASK(5, 4)
|
||||
@@ -303,6 +312,7 @@ struct mtk_wdma_desc {
|
||||
|
||||
#define MTK_WED_WPDMA_RX_D_RST_IDX 0x760
|
||||
#define MTK_WED_WPDMA_RX_D_RST_CRX_IDX GENMASK(17, 16)
|
||||
+#define MTK_WED_WPDMA_RX_D_RST_DRV_IDX_ALL BIT(20)
|
||||
#define MTK_WED_WPDMA_RX_D_RST_DRV_IDX GENMASK(25, 24)
|
||||
|
||||
#define MTK_WED_WPDMA_RX_GLO_CFG 0x76c
|
||||
@@ -313,6 +323,7 @@ struct mtk_wdma_desc {
|
||||
|
||||
#define MTK_WED_WPDMA_RX_D_PREF_CFG 0x7b4
|
||||
#define MTK_WED_WPDMA_RX_D_PREF_EN BIT(0)
|
||||
+#define MTK_WED_WPDMA_RX_D_PREF_BUSY BIT(1)
|
||||
#define MTK_WED_WPDMA_RX_D_PREF_BURST_SIZE GENMASK(12, 8)
|
||||
#define MTK_WED_WPDMA_RX_D_PREF_LOW_THRES GENMASK(21, 16)
|
||||
|
||||
@@ -334,11 +345,13 @@ struct mtk_wdma_desc {
|
||||
|
||||
#define MTK_WED_WDMA_RX_PREF_CFG 0x950
|
||||
#define MTK_WED_WDMA_RX_PREF_EN BIT(0)
|
||||
+#define MTK_WED_WDMA_RX_PREF_BUSY BIT(1)
|
||||
#define MTK_WED_WDMA_RX_PREF_BURST_SIZE GENMASK(12, 8)
|
||||
#define MTK_WED_WDMA_RX_PREF_LOW_THRES GENMASK(21, 16)
|
||||
#define MTK_WED_WDMA_RX_PREF_RX0_SIDX_CLR BIT(24)
|
||||
#define MTK_WED_WDMA_RX_PREF_RX1_SIDX_CLR BIT(25)
|
||||
#define MTK_WED_WDMA_RX_PREF_DDONE2_EN BIT(26)
|
||||
+#define MTK_WED_WDMA_RX_PREF_DDONE2_BUSY BIT(27)
|
||||
|
||||
#define MTK_WED_WDMA_RX_PREF_FIFO_CFG 0x95C
|
||||
#define MTK_WED_WDMA_RX_PREF_FIFO_RX0_CLR BIT(0)
|
||||
@@ -367,6 +380,7 @@ struct mtk_wdma_desc {
|
||||
|
||||
#define MTK_WED_WDMA_RESET_IDX 0xa08
|
||||
#define MTK_WED_WDMA_RESET_IDX_RX GENMASK(17, 16)
|
||||
+#define MTK_WED_WDMA_RESET_IDX_RX_ALL BIT(20)
|
||||
#define MTK_WED_WDMA_RESET_IDX_DRV GENMASK(25, 24)
|
||||
|
||||
#define MTK_WED_WDMA_INT_CLR 0xa24
|
||||
@@ -437,21 +451,62 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WDMA_INT_MASK_RX_DELAY BIT(30)
|
||||
#define MTK_WDMA_INT_MASK_RX_COHERENT BIT(31)
|
||||
|
||||
+#define MTK_WDMA_XDMA_TX_FIFO_CFG 0x238
|
||||
+#define MTK_WDMA_XDMA_TX_FIFO_CFG_TX_PAR_FIFO_CLEAR BIT(0)
|
||||
+#define MTK_WDMA_XDMA_TX_FIFO_CFG_TX_CMD_FIFO_CLEAR BIT(4)
|
||||
+#define MTK_WDMA_XDMA_TX_FIFO_CFG_TX_DMAD_FIFO_CLEAR BIT(8)
|
||||
+#define MTK_WDMA_XDMA_TX_FIFO_CFG_TX_ARR_FIFO_CLEAR BIT(12)
|
||||
+
|
||||
+#define MTK_WDMA_XDMA_RX_FIFO_CFG 0x23c
|
||||
+#define MTK_WDMA_XDMA_RX_FIFO_CFG_RX_PAR_FIFO_CLEAR BIT(0)
|
||||
+#define MTK_WDMA_XDMA_RX_FIFO_CFG_RX_CMD_FIFO_CLEAR BIT(4)
|
||||
+#define MTK_WDMA_XDMA_RX_FIFO_CFG_RX_DMAD_FIFO_CLEAR BIT(8)
|
||||
+#define MTK_WDMA_XDMA_RX_FIFO_CFG_RX_ARR_FIFO_CLEAR BIT(12)
|
||||
+#define MTK_WDMA_XDMA_RX_FIFO_CFG_RX_LEN_FIFO_CLEAR BIT(15)
|
||||
+#define MTK_WDMA_XDMA_RX_FIFO_CFG_RX_WID_FIFO_CLEAR BIT(18)
|
||||
+#define MTK_WDMA_XDMA_RX_FIFO_CFG_RX_BID_FIFO_CLEAR BIT(21)
|
||||
+
|
||||
#define MTK_WDMA_INT_GRP1 0x250
|
||||
#define MTK_WDMA_INT_GRP2 0x254
|
||||
|
||||
#define MTK_WDMA_PREF_TX_CFG 0x2d0
|
||||
#define MTK_WDMA_PREF_TX_CFG_PREF_EN BIT(0)
|
||||
+#define MTK_WDMA_PREF_TX_CFG_PREF_BUSY BIT(1)
|
||||
|
||||
#define MTK_WDMA_PREF_RX_CFG 0x2dc
|
||||
#define MTK_WDMA_PREF_RX_CFG_PREF_EN BIT(0)
|
||||
+#define MTK_WDMA_PREF_RX_CFG_PREF_BUSY BIT(1)
|
||||
+
|
||||
+#define MTK_WDMA_PREF_RX_FIFO_CFG 0x2e0
|
||||
+#define MTK_WDMA_PREF_RX_FIFO_CFG_RING0_CLEAR BIT(0)
|
||||
+#define MTK_WDMA_PREF_RX_FIFO_CFG_RING1_CLEAR BIT(16)
|
||||
+
|
||||
+#define MTK_WDMA_PREF_TX_FIFO_CFG 0x2d4
|
||||
+#define MTK_WDMA_PREF_TX_FIFO_CFG_RING0_CLEAR BIT(0)
|
||||
+#define MTK_WDMA_PREF_TX_FIFO_CFG_RING1_CLEAR BIT(16)
|
||||
+
|
||||
+#define MTK_WDMA_PREF_SIDX_CFG 0x2e4
|
||||
+#define MTK_WDMA_PREF_SIDX_CFG_TX_RING_CLEAR GENMASK(3, 0)
|
||||
+#define MTK_WDMA_PREF_SIDX_CFG_RX_RING_CLEAR GENMASK(5, 4)
|
||||
|
||||
#define MTK_WDMA_WRBK_TX_CFG 0x300
|
||||
+#define MTK_WDMA_WRBK_TX_CFG_WRBK_BUSY BIT(0)
|
||||
#define MTK_WDMA_WRBK_TX_CFG_WRBK_EN BIT(30)
|
||||
|
||||
+#define MTK_WDMA_WRBK_TX_FIFO_CFG(_n) (0x304 + (_n) * 0x4)
|
||||
+#define MTK_WDMA_WRBK_TX_FIFO_CFG_RING_CLEAR BIT(0)
|
||||
+
|
||||
#define MTK_WDMA_WRBK_RX_CFG 0x344
|
||||
+#define MTK_WDMA_WRBK_RX_CFG_WRBK_BUSY BIT(0)
|
||||
#define MTK_WDMA_WRBK_RX_CFG_WRBK_EN BIT(30)
|
||||
|
||||
+#define MTK_WDMA_WRBK_RX_FIFO_CFG(_n) (0x348 + (_n) * 0x4)
|
||||
+#define MTK_WDMA_WRBK_RX_FIFO_CFG_RING_CLEAR BIT(0)
|
||||
+
|
||||
+#define MTK_WDMA_WRBK_SIDX_CFG 0x388
|
||||
+#define MTK_WDMA_WRBK_SIDX_CFG_TX_RING_CLEAR GENMASK(3, 0)
|
||||
+#define MTK_WDMA_WRBK_SIDX_CFG_RX_RING_CLEAR GENMASK(5, 4)
|
||||
+
|
||||
#define MTK_PCIE_MIRROR_MAP(n) ((n) ? 0x4 : 0x0)
|
||||
#define MTK_PCIE_MIRROR_MAP_EN BIT(0)
|
||||
#define MTK_PCIE_MIRROR_MAP_WED_ID BIT(1)
|
||||
@@ -465,6 +520,8 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_RTQM_Q_DBG_BYPASS BIT(5)
|
||||
#define MTK_WED_RTQM_TXDMAD_FPORT GENMASK(23, 20)
|
||||
|
||||
+#define MTK_WED_RTQM_RST 0xb04
|
||||
+
|
||||
#define MTK_WED_RTQM_IGRS0_I2HW_DMAD_CNT 0xb1c
|
||||
#define MTK_WED_RTQM_IGRS0_I2H_DMAD_CNT(_n) (0xb20 + (_n) * 0x4)
|
||||
#define MTK_WED_RTQM_IGRS0_I2HW_PKT_CNT 0xb28
|
||||
@@ -653,6 +710,9 @@ struct mtk_wdma_desc {
|
||||
#define MTK_WED_WPDMA_INT_CTRL_RRO_PG2_CLR BIT(17)
|
||||
#define MTK_WED_WPDMA_INT_CTRL_RRO_PG2_DONE_TRIG GENMASK(22, 18)
|
||||
|
||||
+#define MTK_WED_RRO_RX_HW_STS 0xf00
|
||||
+#define MTK_WED_RX_IND_CMD_BUSY GENMASK(31, 0)
|
||||
+
|
||||
#define MTK_WED_RX_IND_CMD_CNT0 0xf20
|
||||
#define MTK_WED_RX_IND_CMD_DBG_CNT_EN BIT(31)
|
||||
|
@ -0,0 +1,102 @@
|
||||
From 038ba1dc4e54d51d953f5618d8eb5dd39bd9de25 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Tue, 13 Feb 2024 14:35:51 +0100
|
||||
Subject: [PATCH] net: phy: aquantia: add AQR111 and AQR111B0 PHY ID
|
||||
|
||||
Add Aquantia AQR111 and AQR111B0 PHY ID. These PHY advertise 10G speed
|
||||
but actually supports up to 5G speed, hence some manual fixup is needed.
|
||||
|
||||
The Aquantia AQR111B0 PHY is just a variant of the AQR111 with smaller
|
||||
chip size.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Link: https://lore.kernel.org/r/20240213133558.1836-1-ansuelsmth@gmail.com
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
drivers/net/phy/aquantia/aquantia_main.c | 52 ++++++++++++++++++++++++
|
||||
1 file changed, 52 insertions(+)
|
||||
|
||||
--- a/drivers/net/phy/aquantia/aquantia_main.c
|
||||
+++ b/drivers/net/phy/aquantia/aquantia_main.c
|
||||
@@ -22,6 +22,8 @@
|
||||
#define PHY_ID_AQR107 0x03a1b4e0
|
||||
#define PHY_ID_AQCS109 0x03a1b5c2
|
||||
#define PHY_ID_AQR405 0x03a1b4b0
|
||||
+#define PHY_ID_AQR111 0x03a1b610
|
||||
+#define PHY_ID_AQR111B0 0x03a1b612
|
||||
#define PHY_ID_AQR112 0x03a1b662
|
||||
#define PHY_ID_AQR412 0x03a1b712
|
||||
#define PHY_ID_AQR113C 0x31c31c12
|
||||
@@ -672,6 +674,16 @@ static int aqr107_probe(struct phy_devic
|
||||
return aqr_hwmon_probe(phydev);
|
||||
}
|
||||
|
||||
+static int aqr111_config_init(struct phy_device *phydev)
|
||||
+{
|
||||
+ /* AQR111 reports supporting speed up to 10G,
|
||||
+ * however only speeds up to 5G are supported.
|
||||
+ */
|
||||
+ phy_set_max_speed(phydev, SPEED_5000);
|
||||
+
|
||||
+ return aqr107_config_init(phydev);
|
||||
+}
|
||||
+
|
||||
static struct phy_driver aqr_driver[] = {
|
||||
{
|
||||
PHY_ID_MATCH_MODEL(PHY_ID_AQ1202),
|
||||
@@ -746,6 +758,44 @@ static struct phy_driver aqr_driver[] =
|
||||
.link_change_notify = aqr107_link_change_notify,
|
||||
},
|
||||
{
|
||||
+ PHY_ID_MATCH_MODEL(PHY_ID_AQR111),
|
||||
+ .name = "Aquantia AQR111",
|
||||
+ .probe = aqr107_probe,
|
||||
+ .get_rate_matching = aqr107_get_rate_matching,
|
||||
+ .config_init = aqr111_config_init,
|
||||
+ .config_aneg = aqr_config_aneg,
|
||||
+ .config_intr = aqr_config_intr,
|
||||
+ .handle_interrupt = aqr_handle_interrupt,
|
||||
+ .read_status = aqr107_read_status,
|
||||
+ .get_tunable = aqr107_get_tunable,
|
||||
+ .set_tunable = aqr107_set_tunable,
|
||||
+ .suspend = aqr107_suspend,
|
||||
+ .resume = aqr107_resume,
|
||||
+ .get_sset_count = aqr107_get_sset_count,
|
||||
+ .get_strings = aqr107_get_strings,
|
||||
+ .get_stats = aqr107_get_stats,
|
||||
+ .link_change_notify = aqr107_link_change_notify,
|
||||
+},
|
||||
+{
|
||||
+ PHY_ID_MATCH_MODEL(PHY_ID_AQR111B0),
|
||||
+ .name = "Aquantia AQR111B0",
|
||||
+ .probe = aqr107_probe,
|
||||
+ .get_rate_matching = aqr107_get_rate_matching,
|
||||
+ .config_init = aqr111_config_init,
|
||||
+ .config_aneg = aqr_config_aneg,
|
||||
+ .config_intr = aqr_config_intr,
|
||||
+ .handle_interrupt = aqr_handle_interrupt,
|
||||
+ .read_status = aqr107_read_status,
|
||||
+ .get_tunable = aqr107_get_tunable,
|
||||
+ .set_tunable = aqr107_set_tunable,
|
||||
+ .suspend = aqr107_suspend,
|
||||
+ .resume = aqr107_resume,
|
||||
+ .get_sset_count = aqr107_get_sset_count,
|
||||
+ .get_strings = aqr107_get_strings,
|
||||
+ .get_stats = aqr107_get_stats,
|
||||
+ .link_change_notify = aqr107_link_change_notify,
|
||||
+},
|
||||
+{
|
||||
PHY_ID_MATCH_MODEL(PHY_ID_AQR405),
|
||||
.name = "Aquantia AQR405",
|
||||
.config_aneg = aqr_config_aneg,
|
||||
@@ -820,6 +870,8 @@ static struct mdio_device_id __maybe_unu
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR107) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQCS109) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR405) },
|
||||
+ { PHY_ID_MATCH_MODEL(PHY_ID_AQR111) },
|
||||
+ { PHY_ID_MATCH_MODEL(PHY_ID_AQR111B0) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR112) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR412) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR113C) },
|
@ -0,0 +1,60 @@
|
||||
From 71b605d32017e5b8d257db7344bc2f8e8fcc973e Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 15 Feb 2024 16:30:05 +0100
|
||||
Subject: [PATCH] net: phy: aquantia: add AQR113 PHY ID
|
||||
|
||||
Add Aquantia AQR113 PHY ID. Aquantia AQR113 is just a chip size variant of
|
||||
the already supported AQR133C where the only difference is the PHY ID
|
||||
and the hw chip size.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/aquantia/aquantia_main.c | 21 +++++++++++++++++++++
|
||||
1 file changed, 21 insertions(+)
|
||||
|
||||
--- a/drivers/net/phy/aquantia/aquantia_main.c
|
||||
+++ b/drivers/net/phy/aquantia/aquantia_main.c
|
||||
@@ -26,6 +26,7 @@
|
||||
#define PHY_ID_AQR111B0 0x03a1b612
|
||||
#define PHY_ID_AQR112 0x03a1b662
|
||||
#define PHY_ID_AQR412 0x03a1b712
|
||||
+#define PHY_ID_AQR113 0x31c31c40
|
||||
#define PHY_ID_AQR113C 0x31c31c12
|
||||
|
||||
#define MDIO_PHYXS_VEND_IF_STATUS 0xe812
|
||||
@@ -840,6 +841,25 @@ static struct phy_driver aqr_driver[] =
|
||||
.link_change_notify = aqr107_link_change_notify,
|
||||
},
|
||||
{
|
||||
+ PHY_ID_MATCH_MODEL(PHY_ID_AQR113),
|
||||
+ .name = "Aquantia AQR113",
|
||||
+ .probe = aqr107_probe,
|
||||
+ .get_rate_matching = aqr107_get_rate_matching,
|
||||
+ .config_init = aqr107_config_init,
|
||||
+ .config_aneg = aqr_config_aneg,
|
||||
+ .config_intr = aqr_config_intr,
|
||||
+ .handle_interrupt = aqr_handle_interrupt,
|
||||
+ .read_status = aqr107_read_status,
|
||||
+ .get_tunable = aqr107_get_tunable,
|
||||
+ .set_tunable = aqr107_set_tunable,
|
||||
+ .suspend = aqr107_suspend,
|
||||
+ .resume = aqr107_resume,
|
||||
+ .get_sset_count = aqr107_get_sset_count,
|
||||
+ .get_strings = aqr107_get_strings,
|
||||
+ .get_stats = aqr107_get_stats,
|
||||
+ .link_change_notify = aqr107_link_change_notify,
|
||||
+},
|
||||
+{
|
||||
PHY_ID_MATCH_MODEL(PHY_ID_AQR113C),
|
||||
.name = "Aquantia AQR113C",
|
||||
.probe = aqr107_probe,
|
||||
@@ -874,6 +894,7 @@ static struct mdio_device_id __maybe_unu
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR111B0) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR112) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR412) },
|
||||
+ { PHY_ID_MATCH_MODEL(PHY_ID_AQR113) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR113C) },
|
||||
{ }
|
||||
};
|
@ -0,0 +1,59 @@
|
||||
From 6d47302a3f0ba31445478d518d98bd55918bc8ab Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Thu, 15 Feb 2024 22:43:30 +0100
|
||||
Subject: [PATCH] net: phy: aquantia: add AQR813 PHY ID
|
||||
|
||||
Aquantia AQR813 is the Octal Port variant of the AQR113. Add PHY ID for
|
||||
it to provide support for it.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
|
||||
Signed-off-by: David S. Miller <davem@davemloft.net>
|
||||
---
|
||||
drivers/net/phy/aquantia/aquantia_main.c | 21 +++++++++++++++++++++
|
||||
1 file changed, 21 insertions(+)
|
||||
|
||||
--- a/drivers/net/phy/aquantia/aquantia_main.c
|
||||
+++ b/drivers/net/phy/aquantia/aquantia_main.c
|
||||
@@ -28,6 +28,7 @@
|
||||
#define PHY_ID_AQR412 0x03a1b712
|
||||
#define PHY_ID_AQR113 0x31c31c40
|
||||
#define PHY_ID_AQR113C 0x31c31c12
|
||||
+#define PHY_ID_AQR813 0x31c31cb2
|
||||
|
||||
#define MDIO_PHYXS_VEND_IF_STATUS 0xe812
|
||||
#define MDIO_PHYXS_VEND_IF_STATUS_TYPE_MASK GENMASK(7, 3)
|
||||
@@ -878,6 +879,25 @@ static struct phy_driver aqr_driver[] =
|
||||
.get_stats = aqr107_get_stats,
|
||||
.link_change_notify = aqr107_link_change_notify,
|
||||
},
|
||||
+{
|
||||
+ PHY_ID_MATCH_MODEL(PHY_ID_AQR813),
|
||||
+ .name = "Aquantia AQR813",
|
||||
+ .probe = aqr107_probe,
|
||||
+ .get_rate_matching = aqr107_get_rate_matching,
|
||||
+ .config_init = aqr107_config_init,
|
||||
+ .config_aneg = aqr_config_aneg,
|
||||
+ .config_intr = aqr_config_intr,
|
||||
+ .handle_interrupt = aqr_handle_interrupt,
|
||||
+ .read_status = aqr107_read_status,
|
||||
+ .get_tunable = aqr107_get_tunable,
|
||||
+ .set_tunable = aqr107_set_tunable,
|
||||
+ .suspend = aqr107_suspend,
|
||||
+ .resume = aqr107_resume,
|
||||
+ .get_sset_count = aqr107_get_sset_count,
|
||||
+ .get_strings = aqr107_get_strings,
|
||||
+ .get_stats = aqr107_get_stats,
|
||||
+ .link_change_notify = aqr107_link_change_notify,
|
||||
+},
|
||||
};
|
||||
|
||||
module_phy_driver(aqr_driver);
|
||||
@@ -896,6 +916,7 @@ static struct mdio_device_id __maybe_unu
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR412) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR113) },
|
||||
{ PHY_ID_MATCH_MODEL(PHY_ID_AQR113C) },
|
||||
+ { PHY_ID_MATCH_MODEL(PHY_ID_AQR813) },
|
||||
{ }
|
||||
};
|
||||
|
@ -0,0 +1,96 @@
|
||||
From 7f3eb2174512fe6c9c0f062e96eccb0d3cc6d5cd Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Wed, 18 Oct 2023 14:35:47 +0200
|
||||
Subject: [PATCH] net: introduce napi_is_scheduled helper
|
||||
|
||||
We currently have napi_if_scheduled_mark_missed that can be used to
|
||||
check if napi is scheduled but that does more thing than simply checking
|
||||
it and return a bool. Some driver already implement custom function to
|
||||
check if napi is scheduled.
|
||||
|
||||
Drop these custom function and introduce napi_is_scheduled that simply
|
||||
check if napi is scheduled atomically.
|
||||
|
||||
Update any driver and code that implement a similar check and instead
|
||||
use this new helper.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
drivers/net/ethernet/chelsio/cxgb3/sge.c | 8 --------
|
||||
drivers/net/wireless/realtek/rtw89/core.c | 2 +-
|
||||
include/linux/netdevice.h | 23 +++++++++++++++++++++++
|
||||
net/core/dev.c | 2 +-
|
||||
4 files changed, 25 insertions(+), 10 deletions(-)
|
||||
|
||||
--- a/drivers/net/ethernet/chelsio/cxgb3/sge.c
|
||||
+++ b/drivers/net/ethernet/chelsio/cxgb3/sge.c
|
||||
@@ -2501,14 +2501,6 @@ static int napi_rx_handler(struct napi_s
|
||||
return work_done;
|
||||
}
|
||||
|
||||
-/*
|
||||
- * Returns true if the device is already scheduled for polling.
|
||||
- */
|
||||
-static inline int napi_is_scheduled(struct napi_struct *napi)
|
||||
-{
|
||||
- return test_bit(NAPI_STATE_SCHED, &napi->state);
|
||||
-}
|
||||
-
|
||||
/**
|
||||
* process_pure_responses - process pure responses from a response queue
|
||||
* @adap: the adapter
|
||||
--- a/drivers/net/wireless/realtek/rtw89/core.c
|
||||
+++ b/drivers/net/wireless/realtek/rtw89/core.c
|
||||
@@ -1744,7 +1744,7 @@ static void rtw89_core_rx_to_mac80211(st
|
||||
struct napi_struct *napi = &rtwdev->napi;
|
||||
|
||||
/* In low power mode, napi isn't scheduled. Receive it to netif. */
|
||||
- if (unlikely(!test_bit(NAPI_STATE_SCHED, &napi->state)))
|
||||
+ if (unlikely(!napi_is_scheduled(napi)))
|
||||
napi = NULL;
|
||||
|
||||
rtw89_core_hw_to_sband_rate(rx_status);
|
||||
--- a/include/linux/netdevice.h
|
||||
+++ b/include/linux/netdevice.h
|
||||
@@ -480,6 +480,29 @@ static inline bool napi_prefer_busy_poll
|
||||
return test_bit(NAPI_STATE_PREFER_BUSY_POLL, &n->state);
|
||||
}
|
||||
|
||||
+/**
|
||||
+ * napi_is_scheduled - test if NAPI is scheduled
|
||||
+ * @n: NAPI context
|
||||
+ *
|
||||
+ * This check is "best-effort". With no locking implemented,
|
||||
+ * a NAPI can be scheduled or terminate right after this check
|
||||
+ * and produce not precise results.
|
||||
+ *
|
||||
+ * NAPI_STATE_SCHED is an internal state, napi_is_scheduled
|
||||
+ * should not be used normally and napi_schedule should be
|
||||
+ * used instead.
|
||||
+ *
|
||||
+ * Use only if the driver really needs to check if a NAPI
|
||||
+ * is scheduled for example in the context of delayed timer
|
||||
+ * that can be skipped if a NAPI is already scheduled.
|
||||
+ *
|
||||
+ * Return True if NAPI is scheduled, False otherwise.
|
||||
+ */
|
||||
+static inline bool napi_is_scheduled(struct napi_struct *n)
|
||||
+{
|
||||
+ return test_bit(NAPI_STATE_SCHED, &n->state);
|
||||
+}
|
||||
+
|
||||
bool napi_schedule_prep(struct napi_struct *n);
|
||||
|
||||
/**
|
||||
--- a/net/core/dev.c
|
||||
+++ b/net/core/dev.c
|
||||
@@ -6555,7 +6555,7 @@ static int __napi_poll(struct napi_struc
|
||||
* accidentally calling ->poll() when NAPI is not scheduled.
|
||||
*/
|
||||
work = 0;
|
||||
- if (test_bit(NAPI_STATE_SCHED, &n->state)) {
|
||||
+ if (napi_is_scheduled(n)) {
|
||||
work = n->poll(n, weight);
|
||||
trace_napi_poll(n, work, weight);
|
||||
}
|
@ -0,0 +1,77 @@
|
||||
From 2d1a42cf7f77cda54dbbee18d00b1200e7bc22aa Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Wed, 18 Oct 2023 14:35:48 +0200
|
||||
Subject: [PATCH 1/3] net: stmmac: improve TX timer arm logic
|
||||
|
||||
There is currently a problem with the TX timer getting armed multiple
|
||||
unnecessary times causing big performance regression on some device that
|
||||
suffer from heavy handling of hrtimer rearm.
|
||||
|
||||
The use of the TX timer is an old implementation that predates the napi
|
||||
implementation and the interrupt enable/disable handling.
|
||||
|
||||
Due to stmmac being a very old code, the TX timer was never evaluated
|
||||
again with this new implementation and was kept there causing
|
||||
performance regression. The performance regression started to appear
|
||||
with kernel version 4.19 with 8fce33317023 ("net: stmmac: Rework coalesce
|
||||
timer and fix multi-queue races") where the timer was reduced to 1ms
|
||||
causing it to be armed 40 times more than before.
|
||||
|
||||
Decreasing the timer made the problem more present and caused the
|
||||
regression in the other of 600-700mbps on some device (regression where
|
||||
this was notice is ipq806x).
|
||||
|
||||
The problem is in the fact that handling the hrtimer on some target is
|
||||
expensive and recent kernel made the timer armed much more times.
|
||||
A solution that was proposed was reverting the hrtimer change and use
|
||||
mod_timer but such solution would still hide the real problem in the
|
||||
current implementation.
|
||||
|
||||
To fix the regression, apply some additional logic and skip arming the
|
||||
timer when not needed.
|
||||
|
||||
Arm the timer ONLY if a napi is not already scheduled. Running the timer
|
||||
is redundant since the same function (stmmac_tx_clean) will run in the
|
||||
napi TX poll. Also try to cancel any timer if a napi is scheduled to
|
||||
prevent redundant run of TX call.
|
||||
|
||||
With the following new logic the original performance are restored while
|
||||
keeping using the hrtimer.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
.../net/ethernet/stmicro/stmmac/stmmac_main.c | 18 +++++++++++++++---
|
||||
1 file changed, 15 insertions(+), 3 deletions(-)
|
||||
|
||||
--- a/drivers/net/ethernet/stmicro/stmmac/stmmac_main.c
|
||||
+++ b/drivers/net/ethernet/stmicro/stmmac/stmmac_main.c
|
||||
@@ -3003,13 +3003,25 @@ static void stmmac_tx_timer_arm(struct s
|
||||
{
|
||||
struct stmmac_tx_queue *tx_q = &priv->dma_conf.tx_queue[queue];
|
||||
u32 tx_coal_timer = priv->tx_coal_timer[queue];
|
||||
+ struct stmmac_channel *ch;
|
||||
+ struct napi_struct *napi;
|
||||
|
||||
if (!tx_coal_timer)
|
||||
return;
|
||||
|
||||
- hrtimer_start(&tx_q->txtimer,
|
||||
- STMMAC_COAL_TIMER(tx_coal_timer),
|
||||
- HRTIMER_MODE_REL);
|
||||
+ ch = &priv->channel[tx_q->queue_index];
|
||||
+ napi = tx_q->xsk_pool ? &ch->rxtx_napi : &ch->tx_napi;
|
||||
+
|
||||
+ /* Arm timer only if napi is not already scheduled.
|
||||
+ * Try to cancel any timer if napi is scheduled, timer will be armed
|
||||
+ * again in the next scheduled napi.
|
||||
+ */
|
||||
+ if (unlikely(!napi_is_scheduled(napi)))
|
||||
+ hrtimer_start(&tx_q->txtimer,
|
||||
+ STMMAC_COAL_TIMER(tx_coal_timer),
|
||||
+ HRTIMER_MODE_REL);
|
||||
+ else
|
||||
+ hrtimer_try_to_cancel(&tx_q->txtimer);
|
||||
}
|
||||
|
||||
/**
|
@ -0,0 +1,100 @@
|
||||
From a594166387fe08e6f5a32130c400249a35b298f9 Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Wed, 18 Oct 2023 14:35:49 +0200
|
||||
Subject: [PATCH 2/3] net: stmmac: move TX timer arm after DMA enable
|
||||
|
||||
Move TX timer arm call after DMA interrupt is enabled again.
|
||||
|
||||
The TX timer arm function changed logic and now is skipped if a napi is
|
||||
already scheduled. By moving the TX timer arm call after DMA is enabled,
|
||||
we permit to correctly skip if a DMA interrupt has been fired and a napi
|
||||
has been scheduled again.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
.../net/ethernet/stmicro/stmmac/stmmac_main.c | 22 +++++++++++++++----
|
||||
1 file changed, 18 insertions(+), 4 deletions(-)
|
||||
|
||||
--- a/drivers/net/ethernet/stmicro/stmmac/stmmac_main.c
|
||||
+++ b/drivers/net/ethernet/stmicro/stmmac/stmmac_main.c
|
||||
@@ -2551,9 +2551,13 @@ static void stmmac_bump_dma_threshold(st
|
||||
* @priv: driver private structure
|
||||
* @budget: napi budget limiting this functions packet handling
|
||||
* @queue: TX queue index
|
||||
+ * @pending_packets: signal to arm the TX coal timer
|
||||
* Description: it reclaims the transmit resources after transmission completes.
|
||||
+ * If some packets still needs to be handled, due to TX coalesce, set
|
||||
+ * pending_packets to true to make NAPI arm the TX coal timer.
|
||||
*/
|
||||
-static int stmmac_tx_clean(struct stmmac_priv *priv, int budget, u32 queue)
|
||||
+static int stmmac_tx_clean(struct stmmac_priv *priv, int budget, u32 queue,
|
||||
+ bool *pending_packets)
|
||||
{
|
||||
struct stmmac_tx_queue *tx_q = &priv->dma_conf.tx_queue[queue];
|
||||
struct stmmac_txq_stats *txq_stats = &priv->xstats.txq_stats[queue];
|
||||
@@ -2713,7 +2717,7 @@ static int stmmac_tx_clean(struct stmmac
|
||||
|
||||
/* We still have pending packets, let's call for a new scheduling */
|
||||
if (tx_q->dirty_tx != tx_q->cur_tx)
|
||||
- stmmac_tx_timer_arm(priv, queue);
|
||||
+ *pending_packets = true;
|
||||
|
||||
u64_stats_update_begin(&txq_stats->napi_syncp);
|
||||
u64_stats_add(&txq_stats->napi.tx_packets, tx_packets);
|
||||
@@ -5603,6 +5607,7 @@ static int stmmac_napi_poll_tx(struct na
|
||||
container_of(napi, struct stmmac_channel, tx_napi);
|
||||
struct stmmac_priv *priv = ch->priv_data;
|
||||
struct stmmac_txq_stats *txq_stats;
|
||||
+ bool pending_packets = false;
|
||||
u32 chan = ch->index;
|
||||
int work_done;
|
||||
|
||||
@@ -5611,7 +5616,7 @@ static int stmmac_napi_poll_tx(struct na
|
||||
u64_stats_inc(&txq_stats->napi.poll);
|
||||
u64_stats_update_end(&txq_stats->napi_syncp);
|
||||
|
||||
- work_done = stmmac_tx_clean(priv, budget, chan);
|
||||
+ work_done = stmmac_tx_clean(priv, budget, chan, &pending_packets);
|
||||
work_done = min(work_done, budget);
|
||||
|
||||
if (work_done < budget && napi_complete_done(napi, work_done)) {
|
||||
@@ -5622,6 +5627,10 @@ static int stmmac_napi_poll_tx(struct na
|
||||
spin_unlock_irqrestore(&ch->lock, flags);
|
||||
}
|
||||
|
||||
+ /* TX still have packet to handle, check if we need to arm tx timer */
|
||||
+ if (pending_packets)
|
||||
+ stmmac_tx_timer_arm(priv, chan);
|
||||
+
|
||||
return work_done;
|
||||
}
|
||||
|
||||
@@ -5630,6 +5639,7 @@ static int stmmac_napi_poll_rxtx(struct
|
||||
struct stmmac_channel *ch =
|
||||
container_of(napi, struct stmmac_channel, rxtx_napi);
|
||||
struct stmmac_priv *priv = ch->priv_data;
|
||||
+ bool tx_pending_packets = false;
|
||||
int rx_done, tx_done, rxtx_done;
|
||||
struct stmmac_rxq_stats *rxq_stats;
|
||||
struct stmmac_txq_stats *txq_stats;
|
||||
@@ -5645,7 +5655,7 @@ static int stmmac_napi_poll_rxtx(struct
|
||||
u64_stats_inc(&txq_stats->napi.poll);
|
||||
u64_stats_update_end(&txq_stats->napi_syncp);
|
||||
|
||||
- tx_done = stmmac_tx_clean(priv, budget, chan);
|
||||
+ tx_done = stmmac_tx_clean(priv, budget, chan, &tx_pending_packets);
|
||||
tx_done = min(tx_done, budget);
|
||||
|
||||
rx_done = stmmac_rx_zc(priv, budget, chan);
|
||||
@@ -5670,6 +5680,10 @@ static int stmmac_napi_poll_rxtx(struct
|
||||
spin_unlock_irqrestore(&ch->lock, flags);
|
||||
}
|
||||
|
||||
+ /* TX still have packet to handle, check if we need to arm tx timer */
|
||||
+ if (tx_pending_packets)
|
||||
+ stmmac_tx_timer_arm(priv, chan);
|
||||
+
|
||||
return min(rxtx_done, budget - 1);
|
||||
}
|
||||
|
@ -0,0 +1,38 @@
|
||||
From 039550960a2235cfe2dfaa773df9f98f8da31a0c Mon Sep 17 00:00:00 2001
|
||||
From: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Date: Wed, 18 Oct 2023 14:35:50 +0200
|
||||
Subject: [PATCH 3/3] net: stmmac: increase TX coalesce timer to 5ms
|
||||
|
||||
Commit 8fce33317023 ("net: stmmac: Rework coalesce timer and fix
|
||||
multi-queue races") decreased the TX coalesce timer from 40ms to 1ms.
|
||||
|
||||
This caused some performance regression on some target (regression was
|
||||
reported at least on ipq806x) in the order of 600mbps dropping from
|
||||
gigabit handling to only 200mbps.
|
||||
|
||||
The problem was identified in the TX timer getting armed too much time.
|
||||
While this was fixed and improved in another commit, performance can be
|
||||
improved even further by increasing the timer delay a bit moving from
|
||||
1ms to 5ms.
|
||||
|
||||
The value is a good balance between battery saving by prevending too
|
||||
much interrupt to be generated and permitting good performance for
|
||||
internet oriented devices.
|
||||
|
||||
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
||||
Signed-off-by: Paolo Abeni <pabeni@redhat.com>
|
||||
---
|
||||
drivers/net/ethernet/stmicro/stmmac/common.h | 2 +-
|
||||
1 file changed, 1 insertion(+), 1 deletion(-)
|
||||
|
||||
--- a/drivers/net/ethernet/stmicro/stmmac/common.h
|
||||
+++ b/drivers/net/ethernet/stmicro/stmmac/common.h
|
||||
@@ -318,7 +318,7 @@ struct stmmac_safety_stats {
|
||||
#define MIN_DMA_RIWT 0x10
|
||||
#define DEF_DMA_RIWT 0xa0
|
||||
/* Tx coalesce parameters */
|
||||
-#define STMMAC_COAL_TX_TIMER 1000
|
||||
+#define STMMAC_COAL_TX_TIMER 5000
|
||||
#define STMMAC_MAX_COAL_TX_TICK 100000
|
||||
#define STMMAC_TX_MAX_FRAMES 256
|
||||
#define STMMAC_TX_FRAMES 25
|
@ -0,0 +1,122 @@
|
||||
From 788d30daa8f97f06166b6a63f0e51f2a4c2f036a Mon Sep 17 00:00:00 2001
|
||||
From: Hayes Wang <hayeswang@realtek.com>
|
||||
Date: Tue, 26 Sep 2023 19:17:14 +0800
|
||||
Subject: [PATCH] r8152: use napi_gro_frags
|
||||
|
||||
Use napi_gro_frags() for the skb of fragments when the work_done is less
|
||||
than budget.
|
||||
|
||||
Signed-off-by: Hayes Wang <hayeswang@realtek.com>
|
||||
Link: https://lore.kernel.org/r/20230926111714.9448-434-nic_swsd@realtek.com
|
||||
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
|
||||
---
|
||||
drivers/net/usb/r8152.c | 67 ++++++++++++++++++++++++++++++-----------
|
||||
1 file changed, 50 insertions(+), 17 deletions(-)
|
||||
|
||||
--- a/drivers/net/usb/r8152.c
|
||||
+++ b/drivers/net/usb/r8152.c
|
||||
@@ -2584,8 +2584,9 @@ static int rx_bottom(struct r8152 *tp, i
|
||||
while (urb->actual_length > len_used) {
|
||||
struct net_device *netdev = tp->netdev;
|
||||
struct net_device_stats *stats = &netdev->stats;
|
||||
- unsigned int pkt_len, rx_frag_head_sz;
|
||||
+ unsigned int pkt_len, rx_frag_head_sz, len;
|
||||
struct sk_buff *skb;
|
||||
+ bool use_frags;
|
||||
|
||||
WARN_ON_ONCE(skb_queue_len(&tp->rx_queue) >= 1000);
|
||||
|
||||
@@ -2598,45 +2599,77 @@ static int rx_bottom(struct r8152 *tp, i
|
||||
break;
|
||||
|
||||
pkt_len -= ETH_FCS_LEN;
|
||||
+ len = pkt_len;
|
||||
rx_data += sizeof(struct rx_desc);
|
||||
|
||||
- if (!agg_free || tp->rx_copybreak > pkt_len)
|
||||
- rx_frag_head_sz = pkt_len;
|
||||
+ if (!agg_free || tp->rx_copybreak > len)
|
||||
+ use_frags = false;
|
||||
else
|
||||
- rx_frag_head_sz = tp->rx_copybreak;
|
||||
+ use_frags = true;
|
||||
+
|
||||
+ if (use_frags) {
|
||||
+ /* If the budget is exhausted, the packet
|
||||
+ * would be queued in the driver. That is,
|
||||
+ * napi_gro_frags() wouldn't be called, so
|
||||
+ * we couldn't use napi_get_frags().
|
||||
+ */
|
||||
+ if (work_done >= budget) {
|
||||
+ rx_frag_head_sz = tp->rx_copybreak;
|
||||
+ skb = napi_alloc_skb(napi,
|
||||
+ rx_frag_head_sz);
|
||||
+ } else {
|
||||
+ rx_frag_head_sz = 0;
|
||||
+ skb = napi_get_frags(napi);
|
||||
+ }
|
||||
+ } else {
|
||||
+ rx_frag_head_sz = 0;
|
||||
+ skb = napi_alloc_skb(napi, len);
|
||||
+ }
|
||||
|
||||
- skb = napi_alloc_skb(napi, rx_frag_head_sz);
|
||||
if (!skb) {
|
||||
stats->rx_dropped++;
|
||||
goto find_next_rx;
|
||||
}
|
||||
|
||||
skb->ip_summed = r8152_rx_csum(tp, rx_desc);
|
||||
- memcpy(skb->data, rx_data, rx_frag_head_sz);
|
||||
- skb_put(skb, rx_frag_head_sz);
|
||||
- pkt_len -= rx_frag_head_sz;
|
||||
- rx_data += rx_frag_head_sz;
|
||||
- if (pkt_len) {
|
||||
+ rtl_rx_vlan_tag(rx_desc, skb);
|
||||
+
|
||||
+ if (use_frags) {
|
||||
+ if (rx_frag_head_sz) {
|
||||
+ memcpy(skb->data, rx_data,
|
||||
+ rx_frag_head_sz);
|
||||
+ skb_put(skb, rx_frag_head_sz);
|
||||
+ len -= rx_frag_head_sz;
|
||||
+ rx_data += rx_frag_head_sz;
|
||||
+ skb->protocol = eth_type_trans(skb,
|
||||
+ netdev);
|
||||
+ }
|
||||
+
|
||||
skb_add_rx_frag(skb, 0, agg->page,
|
||||
agg_offset(agg, rx_data),
|
||||
- pkt_len,
|
||||
- SKB_DATA_ALIGN(pkt_len));
|
||||
+ len, SKB_DATA_ALIGN(len));
|
||||
get_page(agg->page);
|
||||
+ } else {
|
||||
+ memcpy(skb->data, rx_data, len);
|
||||
+ skb_put(skb, len);
|
||||
+ skb->protocol = eth_type_trans(skb, netdev);
|
||||
}
|
||||
|
||||
- skb->protocol = eth_type_trans(skb, netdev);
|
||||
- rtl_rx_vlan_tag(rx_desc, skb);
|
||||
if (work_done < budget) {
|
||||
+ if (use_frags)
|
||||
+ napi_gro_frags(napi);
|
||||
+ else
|
||||
+ napi_gro_receive(napi, skb);
|
||||
+
|
||||
work_done++;
|
||||
stats->rx_packets++;
|
||||
- stats->rx_bytes += skb->len;
|
||||
- napi_gro_receive(napi, skb);
|
||||
+ stats->rx_bytes += pkt_len;
|
||||
} else {
|
||||
__skb_queue_tail(&tp->rx_queue, skb);
|
||||
}
|
||||
|
||||
find_next_rx:
|
||||
- rx_data = rx_agg_align(rx_data + pkt_len + ETH_FCS_LEN);
|
||||
+ rx_data = rx_agg_align(rx_data + len + ETH_FCS_LEN);
|
||||
rx_desc = (struct rx_desc *)rx_data;
|
||||
len_used = agg_offset(agg, rx_data);
|
||||
len_used += sizeof(struct rx_desc);
|
@ -0,0 +1,207 @@
|
||||
From a64c3c1357275b1fd61bc9734f638cdb5d8a8bbb Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Marek=20Beh=C3=BAn?= <kabel@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 18:11:02 +0200
|
||||
Subject: [PATCH 4/6] leds: turris-omnia: Make set_brightness() more efficient
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
Implement caching of the LED color and state values that are sent to MCU
|
||||
in order to make the set_brightness() operation more efficient by
|
||||
avoiding I2C transactions which are not needed.
|
||||
|
||||
On Turris Omnia's MCU, which acts as the RGB LED controller, each LED
|
||||
has a RGB color, and a ON/OFF state, which are configurable via I2C
|
||||
commands CMD_LED_COLOR and CMD_LED_STATE.
|
||||
|
||||
The CMD_LED_COLOR command sends 5 bytes and the CMD_LED_STATE command 2
|
||||
bytes over the I2C bus, which operates at 100 kHz. With I2C overhead
|
||||
this allows ~1670 color changing commands and ~3200 state changing
|
||||
commands per second (or around 1000 color + state changes per second).
|
||||
This may seem more than enough, but the issue is that the I2C bus is
|
||||
shared with another peripheral, the MCU. The MCU exposes an interrupt
|
||||
interface, and it can trigger hundreds of interrupts per second. Each
|
||||
time, we need to read the interrupt state register over this I2C bus.
|
||||
Whenever we are sending a LED color/state changing command, the
|
||||
interrupt reading is waiting.
|
||||
|
||||
Currently, every time LED brightness or LED multi intensity is changed,
|
||||
we send a CMD_LED_STATE command, and if the computed color (brightness
|
||||
adjusted multi_intensity) is non-zero, we also send a CMD_LED_COLOR
|
||||
command.
|
||||
|
||||
Consider for example the situation when we have a netdev trigger enabled
|
||||
for a LED. The netdev trigger does not change the LED color, only the
|
||||
brightness (either to 0 or to currently configured brightness), and so
|
||||
there is no need to send the CMD_LED_COLOR command. But each change of
|
||||
brightness to 0 sends one CMD_LED_STATE command, and each change of
|
||||
brightness to max_brightness sends one CMD_LED_STATE command and one
|
||||
CMD_LED_COLOR command:
|
||||
set_brightness(0) -> CMD_LED_STATE
|
||||
set_brightness(255) -> CMD_LED_STATE + CMD_LED_COLOR
|
||||
(unnecessary)
|
||||
|
||||
We can avoid the unnecessary I2C transactions if we cache the values of
|
||||
state and color that are sent to the controller. If the color does not
|
||||
change from the one previously sent, there is no need to do the
|
||||
CMD_LED_COLOR I2C transaction, and if the state does not change, there
|
||||
is no need to do the CMD_LED_STATE transaction.
|
||||
|
||||
Because we need to make sure that our cached values are consistent with
|
||||
the controller state, add explicit setting of the LED color to white at
|
||||
probe time (this is the default setting when MCU resets, but does not
|
||||
necessarily need to be the case, for example if U-Boot played with the
|
||||
LED colors).
|
||||
|
||||
Signed-off-by: Marek Behún <kabel@kernel.org>
|
||||
Link: https://lore.kernel.org/r/20230918161104.20860-3-kabel@kernel.org
|
||||
Signed-off-by: Lee Jones <lee@kernel.org>
|
||||
---
|
||||
drivers/leds/leds-turris-omnia.c | 96 ++++++++++++++++++++++++++------
|
||||
1 file changed, 78 insertions(+), 18 deletions(-)
|
||||
|
||||
--- a/drivers/leds/leds-turris-omnia.c
|
||||
+++ b/drivers/leds/leds-turris-omnia.c
|
||||
@@ -30,6 +30,8 @@
|
||||
struct omnia_led {
|
||||
struct led_classdev_mc mc_cdev;
|
||||
struct mc_subled subled_info[OMNIA_LED_NUM_CHANNELS];
|
||||
+ u8 cached_channels[OMNIA_LED_NUM_CHANNELS];
|
||||
+ bool on;
|
||||
int reg;
|
||||
};
|
||||
|
||||
@@ -72,36 +74,82 @@ static int omnia_cmd_read_u8(const struc
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
+static int omnia_led_send_color_cmd(const struct i2c_client *client,
|
||||
+ struct omnia_led *led)
|
||||
+{
|
||||
+ char cmd[5];
|
||||
+ int ret;
|
||||
+
|
||||
+ cmd[0] = CMD_LED_COLOR;
|
||||
+ cmd[1] = led->reg;
|
||||
+ cmd[2] = led->subled_info[0].brightness;
|
||||
+ cmd[3] = led->subled_info[1].brightness;
|
||||
+ cmd[4] = led->subled_info[2].brightness;
|
||||
+
|
||||
+ /* Send the color change command */
|
||||
+ ret = i2c_master_send(client, cmd, 5);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+
|
||||
+ /* Cache the RGB channel brightnesses */
|
||||
+ for (int i = 0; i < OMNIA_LED_NUM_CHANNELS; ++i)
|
||||
+ led->cached_channels[i] = led->subled_info[i].brightness;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+/* Determine if the computed RGB channels are different from the cached ones */
|
||||
+static bool omnia_led_channels_changed(struct omnia_led *led)
|
||||
+{
|
||||
+ for (int i = 0; i < OMNIA_LED_NUM_CHANNELS; ++i)
|
||||
+ if (led->subled_info[i].brightness != led->cached_channels[i])
|
||||
+ return true;
|
||||
+
|
||||
+ return false;
|
||||
+}
|
||||
+
|
||||
static int omnia_led_brightness_set_blocking(struct led_classdev *cdev,
|
||||
enum led_brightness brightness)
|
||||
{
|
||||
struct led_classdev_mc *mc_cdev = lcdev_to_mccdev(cdev);
|
||||
struct omnia_leds *leds = dev_get_drvdata(cdev->dev->parent);
|
||||
struct omnia_led *led = to_omnia_led(mc_cdev);
|
||||
- u8 buf[5], state;
|
||||
- int ret;
|
||||
+ int err = 0;
|
||||
|
||||
mutex_lock(&leds->lock);
|
||||
|
||||
- led_mc_calc_color_components(&led->mc_cdev, brightness);
|
||||
+ /*
|
||||
+ * Only recalculate RGB brightnesses from intensities if brightness is
|
||||
+ * non-zero. Otherwise we won't be using them and we can save ourselves
|
||||
+ * some software divisions (Omnia's CPU does not implement the division
|
||||
+ * instruction).
|
||||
+ */
|
||||
+ if (brightness) {
|
||||
+ led_mc_calc_color_components(mc_cdev, brightness);
|
||||
+
|
||||
+ /*
|
||||
+ * Send color command only if brightness is non-zero and the RGB
|
||||
+ * channel brightnesses changed.
|
||||
+ */
|
||||
+ if (omnia_led_channels_changed(led))
|
||||
+ err = omnia_led_send_color_cmd(leds->client, led);
|
||||
+ }
|
||||
|
||||
- buf[0] = CMD_LED_COLOR;
|
||||
- buf[1] = led->reg;
|
||||
- buf[2] = mc_cdev->subled_info[0].brightness;
|
||||
- buf[3] = mc_cdev->subled_info[1].brightness;
|
||||
- buf[4] = mc_cdev->subled_info[2].brightness;
|
||||
-
|
||||
- state = CMD_LED_STATE_LED(led->reg);
|
||||
- if (buf[2] || buf[3] || buf[4])
|
||||
- state |= CMD_LED_STATE_ON;
|
||||
-
|
||||
- ret = omnia_cmd_write_u8(leds->client, CMD_LED_STATE, state);
|
||||
- if (ret >= 0 && (state & CMD_LED_STATE_ON))
|
||||
- ret = i2c_master_send(leds->client, buf, 5);
|
||||
+ /* Send on/off state change only if (bool)brightness changed */
|
||||
+ if (!err && !brightness != !led->on) {
|
||||
+ u8 state = CMD_LED_STATE_LED(led->reg);
|
||||
+
|
||||
+ if (brightness)
|
||||
+ state |= CMD_LED_STATE_ON;
|
||||
+
|
||||
+ err = omnia_cmd_write_u8(leds->client, CMD_LED_STATE, state);
|
||||
+ if (!err)
|
||||
+ led->on = !!brightness;
|
||||
+ }
|
||||
|
||||
mutex_unlock(&leds->lock);
|
||||
|
||||
- return ret;
|
||||
+ return err;
|
||||
}
|
||||
|
||||
static int omnia_led_register(struct i2c_client *client, struct omnia_led *led,
|
||||
@@ -129,11 +177,15 @@ static int omnia_led_register(struct i2c
|
||||
}
|
||||
|
||||
led->subled_info[0].color_index = LED_COLOR_ID_RED;
|
||||
- led->subled_info[0].channel = 0;
|
||||
led->subled_info[1].color_index = LED_COLOR_ID_GREEN;
|
||||
- led->subled_info[1].channel = 1;
|
||||
led->subled_info[2].color_index = LED_COLOR_ID_BLUE;
|
||||
- led->subled_info[2].channel = 2;
|
||||
+
|
||||
+ /* Initial color is white */
|
||||
+ for (int i = 0; i < OMNIA_LED_NUM_CHANNELS; ++i) {
|
||||
+ led->subled_info[i].intensity = 255;
|
||||
+ led->subled_info[i].brightness = 255;
|
||||
+ led->subled_info[i].channel = i;
|
||||
+ }
|
||||
|
||||
led->mc_cdev.subled_info = led->subled_info;
|
||||
led->mc_cdev.num_colors = OMNIA_LED_NUM_CHANNELS;
|
||||
@@ -162,6 +214,14 @@ static int omnia_led_register(struct i2c
|
||||
return ret;
|
||||
}
|
||||
|
||||
+ /* Set initial color and cache it */
|
||||
+ ret = omnia_led_send_color_cmd(client, led);
|
||||
+ if (ret < 0) {
|
||||
+ dev_err(dev, "Cannot set LED %pOF initial color: %i\n", np,
|
||||
+ ret);
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
ret = devm_led_classdev_multicolor_register_ext(dev, &led->mc_cdev,
|
||||
&init_data);
|
||||
if (ret < 0) {
|
@ -0,0 +1,202 @@
|
||||
From e965e0f6de60874fc0a0caed9a9e0122999e0c7b Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Marek=20Beh=C3=BAn?= <kabel@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 18:11:03 +0200
|
||||
Subject: [PATCH 5/6] leds: turris-omnia: Support HW controlled mode via
|
||||
private trigger
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
Add support for enabling MCU controlled mode of the Turris Omnia LEDs
|
||||
via a LED private trigger called "omnia-mcu". Recall that private LED
|
||||
triggers will only be listed in the sysfs trigger file for LEDs that
|
||||
support them (currently there is no user of this mechanism).
|
||||
|
||||
When in MCU controlled mode, the user can still set LED color, but the
|
||||
blinking is done by MCU, which does different things for different LEDs:
|
||||
- WAN LED is blinked according to the LED[0] pin of the WAN PHY
|
||||
- LAN LEDs are blinked according to the LED[0] output of the
|
||||
corresponding port of the LAN switch
|
||||
- PCIe LEDs are blinked according to the logical OR of the MiniPCIe port
|
||||
LED pins
|
||||
|
||||
In the future I want to make the netdev trigger to transparently offload
|
||||
the blinking to the HW if user sets compatible settings for the netdev
|
||||
trigger (for LEDs associated with network devices).
|
||||
There was some work on this already, and hopefully we will be able to
|
||||
complete it sometime, but for now there are still multiple blockers for
|
||||
this, and even if there weren't, we still would not be able to configure
|
||||
HW controlled mode for the LEDs associated with MiniPCIe ports.
|
||||
|
||||
In the meantime let's support HW controlled mode via the private LED
|
||||
trigger mechanism. If, in the future, we manage to complete the netdev
|
||||
trigger offloading, we can still keep this private trigger for backwards
|
||||
compatibility, if needed.
|
||||
|
||||
We also set "omnia-mcu" to cdev->default_trigger, so that the MCU keeps
|
||||
control until the user first wants to take over it. If a different
|
||||
default trigger is specified in device-tree via the
|
||||
'linux,default-trigger' property, LED class will overwrite
|
||||
cdev->default_trigger, and so the DT property will be respected.
|
||||
|
||||
Signed-off-by: Marek Behún <kabel@kernel.org>
|
||||
Link: https://lore.kernel.org/r/20230918161104.20860-4-kabel@kernel.org
|
||||
Signed-off-by: Lee Jones <lee@kernel.org>
|
||||
---
|
||||
drivers/leds/Kconfig | 1 +
|
||||
drivers/leds/leds-turris-omnia.c | 98 +++++++++++++++++++++++++++++---
|
||||
2 files changed, 91 insertions(+), 8 deletions(-)
|
||||
|
||||
--- a/drivers/leds/Kconfig
|
||||
+++ b/drivers/leds/Kconfig
|
||||
@@ -188,6 +188,7 @@ config LEDS_TURRIS_OMNIA
|
||||
depends on I2C
|
||||
depends on MACH_ARMADA_38X || COMPILE_TEST
|
||||
depends on OF
|
||||
+ select LEDS_TRIGGERS
|
||||
help
|
||||
This option enables basic support for the LEDs found on the front
|
||||
side of CZ.NIC's Turris Omnia router. There are 12 RGB LEDs on the
|
||||
--- a/drivers/leds/leds-turris-omnia.c
|
||||
+++ b/drivers/leds/leds-turris-omnia.c
|
||||
@@ -31,7 +31,7 @@ struct omnia_led {
|
||||
struct led_classdev_mc mc_cdev;
|
||||
struct mc_subled subled_info[OMNIA_LED_NUM_CHANNELS];
|
||||
u8 cached_channels[OMNIA_LED_NUM_CHANNELS];
|
||||
- bool on;
|
||||
+ bool on, hwtrig;
|
||||
int reg;
|
||||
};
|
||||
|
||||
@@ -120,12 +120,14 @@ static int omnia_led_brightness_set_bloc
|
||||
|
||||
/*
|
||||
* Only recalculate RGB brightnesses from intensities if brightness is
|
||||
- * non-zero. Otherwise we won't be using them and we can save ourselves
|
||||
- * some software divisions (Omnia's CPU does not implement the division
|
||||
- * instruction).
|
||||
+ * non-zero (if it is zero and the LED is in HW blinking mode, we use
|
||||
+ * max_brightness as brightness). Otherwise we won't be using them and
|
||||
+ * we can save ourselves some software divisions (Omnia's CPU does not
|
||||
+ * implement the division instruction).
|
||||
*/
|
||||
- if (brightness) {
|
||||
- led_mc_calc_color_components(mc_cdev, brightness);
|
||||
+ if (brightness || led->hwtrig) {
|
||||
+ led_mc_calc_color_components(mc_cdev, brightness ?:
|
||||
+ cdev->max_brightness);
|
||||
|
||||
/*
|
||||
* Send color command only if brightness is non-zero and the RGB
|
||||
@@ -135,8 +137,11 @@ static int omnia_led_brightness_set_bloc
|
||||
err = omnia_led_send_color_cmd(leds->client, led);
|
||||
}
|
||||
|
||||
- /* Send on/off state change only if (bool)brightness changed */
|
||||
- if (!err && !brightness != !led->on) {
|
||||
+ /*
|
||||
+ * Send on/off state change only if (bool)brightness changed and the LED
|
||||
+ * is not being blinked by HW.
|
||||
+ */
|
||||
+ if (!err && !led->hwtrig && !brightness != !led->on) {
|
||||
u8 state = CMD_LED_STATE_LED(led->reg);
|
||||
|
||||
if (brightness)
|
||||
@@ -152,6 +157,71 @@ static int omnia_led_brightness_set_bloc
|
||||
return err;
|
||||
}
|
||||
|
||||
+static struct led_hw_trigger_type omnia_hw_trigger_type;
|
||||
+
|
||||
+static int omnia_hwtrig_activate(struct led_classdev *cdev)
|
||||
+{
|
||||
+ struct led_classdev_mc *mc_cdev = lcdev_to_mccdev(cdev);
|
||||
+ struct omnia_leds *leds = dev_get_drvdata(cdev->dev->parent);
|
||||
+ struct omnia_led *led = to_omnia_led(mc_cdev);
|
||||
+ int err = 0;
|
||||
+
|
||||
+ mutex_lock(&leds->lock);
|
||||
+
|
||||
+ if (!led->on) {
|
||||
+ /*
|
||||
+ * If the LED is off (brightness was set to 0), the last
|
||||
+ * configured color was not necessarily sent to the MCU.
|
||||
+ * Recompute with max_brightness and send if needed.
|
||||
+ */
|
||||
+ led_mc_calc_color_components(mc_cdev, cdev->max_brightness);
|
||||
+
|
||||
+ if (omnia_led_channels_changed(led))
|
||||
+ err = omnia_led_send_color_cmd(leds->client, led);
|
||||
+ }
|
||||
+
|
||||
+ if (!err) {
|
||||
+ /* Put the LED into MCU controlled mode */
|
||||
+ err = omnia_cmd_write_u8(leds->client, CMD_LED_MODE,
|
||||
+ CMD_LED_MODE_LED(led->reg));
|
||||
+ if (!err)
|
||||
+ led->hwtrig = true;
|
||||
+ }
|
||||
+
|
||||
+ mutex_unlock(&leds->lock);
|
||||
+
|
||||
+ return err;
|
||||
+}
|
||||
+
|
||||
+static void omnia_hwtrig_deactivate(struct led_classdev *cdev)
|
||||
+{
|
||||
+ struct omnia_leds *leds = dev_get_drvdata(cdev->dev->parent);
|
||||
+ struct omnia_led *led = to_omnia_led(lcdev_to_mccdev(cdev));
|
||||
+ int err;
|
||||
+
|
||||
+ mutex_lock(&leds->lock);
|
||||
+
|
||||
+ led->hwtrig = false;
|
||||
+
|
||||
+ /* Put the LED into software mode */
|
||||
+ err = omnia_cmd_write_u8(leds->client, CMD_LED_MODE,
|
||||
+ CMD_LED_MODE_LED(led->reg) |
|
||||
+ CMD_LED_MODE_USER);
|
||||
+
|
||||
+ mutex_unlock(&leds->lock);
|
||||
+
|
||||
+ if (err < 0)
|
||||
+ dev_err(cdev->dev, "Cannot put LED to software mode: %i\n",
|
||||
+ err);
|
||||
+}
|
||||
+
|
||||
+static struct led_trigger omnia_hw_trigger = {
|
||||
+ .name = "omnia-mcu",
|
||||
+ .activate = omnia_hwtrig_activate,
|
||||
+ .deactivate = omnia_hwtrig_deactivate,
|
||||
+ .trigger_type = &omnia_hw_trigger_type,
|
||||
+};
|
||||
+
|
||||
static int omnia_led_register(struct i2c_client *client, struct omnia_led *led,
|
||||
struct device_node *np)
|
||||
{
|
||||
@@ -195,6 +265,12 @@ static int omnia_led_register(struct i2c
|
||||
cdev = &led->mc_cdev.led_cdev;
|
||||
cdev->max_brightness = 255;
|
||||
cdev->brightness_set_blocking = omnia_led_brightness_set_blocking;
|
||||
+ cdev->trigger_type = &omnia_hw_trigger_type;
|
||||
+ /*
|
||||
+ * Use the omnia-mcu trigger as the default trigger. It may be rewritten
|
||||
+ * by LED class from the linux,default-trigger property.
|
||||
+ */
|
||||
+ cdev->default_trigger = omnia_hw_trigger.name;
|
||||
|
||||
/* put the LED into software mode */
|
||||
ret = omnia_cmd_write_u8(client, CMD_LED_MODE,
|
||||
@@ -308,6 +384,12 @@ static int omnia_leds_probe(struct i2c_c
|
||||
|
||||
mutex_init(&leds->lock);
|
||||
|
||||
+ ret = devm_led_trigger_register(dev, &omnia_hw_trigger);
|
||||
+ if (ret < 0) {
|
||||
+ dev_err(dev, "Cannot register private LED trigger: %d\n", ret);
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
led = &leds->leds[0];
|
||||
for_each_available_child_of_node(np, child) {
|
||||
ret = omnia_led_register(client, led, child);
|
@ -0,0 +1,244 @@
|
||||
From 0efb3f9609d3de5a7d8c31e3835d7eb3e6adce79 Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Marek=20Beh=C3=BAn?= <kabel@kernel.org>
|
||||
Date: Mon, 18 Sep 2023 18:11:04 +0200
|
||||
Subject: [PATCH 6/6] leds: turris-omnia: Add support for enabling/disabling HW
|
||||
gamma correction
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
If the MCU on Turris Omnia is running newer firmware versions, the LED
|
||||
controller supports RGB gamma correction (and enables it by default for
|
||||
newer boards).
|
||||
|
||||
Determine whether the gamma correction setting feature is supported and
|
||||
add the ability to set it via sysfs attribute file.
|
||||
|
||||
Signed-off-by: Marek Behún <kabel@kernel.org>
|
||||
Link: https://lore.kernel.org/r/20230918161104.20860-5-kabel@kernel.org
|
||||
Signed-off-by: Lee Jones <lee@kernel.org>
|
||||
---
|
||||
.../sysfs-class-led-driver-turris-omnia | 14 ++
|
||||
drivers/leds/leds-turris-omnia.c | 137 +++++++++++++++---
|
||||
2 files changed, 134 insertions(+), 17 deletions(-)
|
||||
|
||||
--- a/Documentation/ABI/testing/sysfs-class-led-driver-turris-omnia
|
||||
+++ b/Documentation/ABI/testing/sysfs-class-led-driver-turris-omnia
|
||||
@@ -12,3 +12,17 @@ Description: (RW) On the front panel of
|
||||
able to change this setting from software.
|
||||
|
||||
Format: %i
|
||||
+
|
||||
+What: /sys/class/leds/<led>/device/gamma_correction
|
||||
+Date: August 2023
|
||||
+KernelVersion: 6.6
|
||||
+Contact: Marek Behún <kabel@kernel.org>
|
||||
+Description: (RW) Newer versions of the microcontroller firmware of the
|
||||
+ Turris Omnia router support gamma correction for the RGB LEDs.
|
||||
+ This feature can be enabled/disabled by writing to this file.
|
||||
+
|
||||
+ If the feature is not supported because the MCU firmware is too
|
||||
+ old, the file always reads as 0, and writing to the file results
|
||||
+ in the EOPNOTSUPP error.
|
||||
+
|
||||
+ Format: %i
|
||||
--- a/drivers/leds/leds-turris-omnia.c
|
||||
+++ b/drivers/leds/leds-turris-omnia.c
|
||||
@@ -15,17 +15,30 @@
|
||||
#define OMNIA_BOARD_LEDS 12
|
||||
#define OMNIA_LED_NUM_CHANNELS 3
|
||||
|
||||
-#define CMD_LED_MODE 3
|
||||
-#define CMD_LED_MODE_LED(l) ((l) & 0x0f)
|
||||
-#define CMD_LED_MODE_USER 0x10
|
||||
-
|
||||
-#define CMD_LED_STATE 4
|
||||
-#define CMD_LED_STATE_LED(l) ((l) & 0x0f)
|
||||
-#define CMD_LED_STATE_ON 0x10
|
||||
-
|
||||
-#define CMD_LED_COLOR 5
|
||||
-#define CMD_LED_SET_BRIGHTNESS 7
|
||||
-#define CMD_LED_GET_BRIGHTNESS 8
|
||||
+/* MCU controller commands at I2C address 0x2a */
|
||||
+#define OMNIA_MCU_I2C_ADDR 0x2a
|
||||
+
|
||||
+#define CMD_GET_STATUS_WORD 0x01
|
||||
+#define STS_FEATURES_SUPPORTED BIT(2)
|
||||
+
|
||||
+#define CMD_GET_FEATURES 0x10
|
||||
+#define FEAT_LED_GAMMA_CORRECTION BIT(5)
|
||||
+
|
||||
+/* LED controller commands at I2C address 0x2b */
|
||||
+#define CMD_LED_MODE 0x03
|
||||
+#define CMD_LED_MODE_LED(l) ((l) & 0x0f)
|
||||
+#define CMD_LED_MODE_USER 0x10
|
||||
+
|
||||
+#define CMD_LED_STATE 0x04
|
||||
+#define CMD_LED_STATE_LED(l) ((l) & 0x0f)
|
||||
+#define CMD_LED_STATE_ON 0x10
|
||||
+
|
||||
+#define CMD_LED_COLOR 0x05
|
||||
+#define CMD_LED_SET_BRIGHTNESS 0x07
|
||||
+#define CMD_LED_GET_BRIGHTNESS 0x08
|
||||
+
|
||||
+#define CMD_SET_GAMMA_CORRECTION 0x30
|
||||
+#define CMD_GET_GAMMA_CORRECTION 0x31
|
||||
|
||||
struct omnia_led {
|
||||
struct led_classdev_mc mc_cdev;
|
||||
@@ -40,6 +53,7 @@ struct omnia_led {
|
||||
struct omnia_leds {
|
||||
struct i2c_client *client;
|
||||
struct mutex lock;
|
||||
+ bool has_gamma_correction;
|
||||
struct omnia_led leds[];
|
||||
};
|
||||
|
||||
@@ -50,30 +64,42 @@ static int omnia_cmd_write_u8(const stru
|
||||
return i2c_master_send(client, buf, sizeof(buf));
|
||||
}
|
||||
|
||||
-static int omnia_cmd_read_u8(const struct i2c_client *client, u8 cmd)
|
||||
+static int omnia_cmd_read_raw(struct i2c_adapter *adapter, u8 addr, u8 cmd,
|
||||
+ void *reply, size_t len)
|
||||
{
|
||||
struct i2c_msg msgs[2];
|
||||
- u8 reply;
|
||||
int ret;
|
||||
|
||||
- msgs[0].addr = client->addr;
|
||||
+ msgs[0].addr = addr;
|
||||
msgs[0].flags = 0;
|
||||
msgs[0].len = 1;
|
||||
msgs[0].buf = &cmd;
|
||||
- msgs[1].addr = client->addr;
|
||||
+ msgs[1].addr = addr;
|
||||
msgs[1].flags = I2C_M_RD;
|
||||
- msgs[1].len = 1;
|
||||
- msgs[1].buf = &reply;
|
||||
+ msgs[1].len = len;
|
||||
+ msgs[1].buf = reply;
|
||||
|
||||
- ret = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
|
||||
+ ret = i2c_transfer(adapter, msgs, ARRAY_SIZE(msgs));
|
||||
if (likely(ret == ARRAY_SIZE(msgs)))
|
||||
- return reply;
|
||||
+ return len;
|
||||
else if (ret < 0)
|
||||
return ret;
|
||||
else
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
+static int omnia_cmd_read_u8(const struct i2c_client *client, u8 cmd)
|
||||
+{
|
||||
+ u8 reply;
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = omnia_cmd_read_raw(client->adapter, client->addr, cmd, &reply, 1);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+
|
||||
+ return reply;
|
||||
+}
|
||||
+
|
||||
static int omnia_led_send_color_cmd(const struct i2c_client *client,
|
||||
struct omnia_led *led)
|
||||
{
|
||||
@@ -352,12 +378,74 @@ static ssize_t brightness_store(struct d
|
||||
}
|
||||
static DEVICE_ATTR_RW(brightness);
|
||||
|
||||
+static ssize_t gamma_correction_show(struct device *dev,
|
||||
+ struct device_attribute *a, char *buf)
|
||||
+{
|
||||
+ struct i2c_client *client = to_i2c_client(dev);
|
||||
+ struct omnia_leds *leds = i2c_get_clientdata(client);
|
||||
+ int ret;
|
||||
+
|
||||
+ if (leds->has_gamma_correction) {
|
||||
+ ret = omnia_cmd_read_u8(client, CMD_GET_GAMMA_CORRECTION);
|
||||
+ if (ret < 0)
|
||||
+ return ret;
|
||||
+ } else {
|
||||
+ ret = 0;
|
||||
+ }
|
||||
+
|
||||
+ return sysfs_emit(buf, "%d\n", !!ret);
|
||||
+}
|
||||
+
|
||||
+static ssize_t gamma_correction_store(struct device *dev,
|
||||
+ struct device_attribute *a,
|
||||
+ const char *buf, size_t count)
|
||||
+{
|
||||
+ struct i2c_client *client = to_i2c_client(dev);
|
||||
+ struct omnia_leds *leds = i2c_get_clientdata(client);
|
||||
+ bool val;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (!leds->has_gamma_correction)
|
||||
+ return -EOPNOTSUPP;
|
||||
+
|
||||
+ if (kstrtobool(buf, &val) < 0)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ ret = omnia_cmd_write_u8(client, CMD_SET_GAMMA_CORRECTION, val);
|
||||
+
|
||||
+ return ret < 0 ? ret : count;
|
||||
+}
|
||||
+static DEVICE_ATTR_RW(gamma_correction);
|
||||
+
|
||||
static struct attribute *omnia_led_controller_attrs[] = {
|
||||
&dev_attr_brightness.attr,
|
||||
+ &dev_attr_gamma_correction.attr,
|
||||
NULL,
|
||||
};
|
||||
ATTRIBUTE_GROUPS(omnia_led_controller);
|
||||
|
||||
+static int omnia_mcu_get_features(const struct i2c_client *client)
|
||||
+{
|
||||
+ u16 reply;
|
||||
+ int err;
|
||||
+
|
||||
+ err = omnia_cmd_read_raw(client->adapter, OMNIA_MCU_I2C_ADDR,
|
||||
+ CMD_GET_STATUS_WORD, &reply, sizeof(reply));
|
||||
+ if (err < 0)
|
||||
+ return err;
|
||||
+
|
||||
+ /* Check whether MCU firmware supports the CMD_GET_FEAUTRES command */
|
||||
+ if (!(le16_to_cpu(reply) & STS_FEATURES_SUPPORTED))
|
||||
+ return 0;
|
||||
+
|
||||
+ err = omnia_cmd_read_raw(client->adapter, OMNIA_MCU_I2C_ADDR,
|
||||
+ CMD_GET_FEATURES, &reply, sizeof(reply));
|
||||
+ if (err < 0)
|
||||
+ return err;
|
||||
+
|
||||
+ return le16_to_cpu(reply);
|
||||
+}
|
||||
+
|
||||
static int omnia_leds_probe(struct i2c_client *client)
|
||||
{
|
||||
struct device *dev = &client->dev;
|
||||
@@ -382,6 +470,21 @@ static int omnia_leds_probe(struct i2c_c
|
||||
leds->client = client;
|
||||
i2c_set_clientdata(client, leds);
|
||||
|
||||
+ ret = omnia_mcu_get_features(client);
|
||||
+ if (ret < 0) {
|
||||
+ dev_err(dev, "Cannot determine MCU supported features: %d\n",
|
||||
+ ret);
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ leds->has_gamma_correction = ret & FEAT_LED_GAMMA_CORRECTION;
|
||||
+ if (!leds->has_gamma_correction) {
|
||||
+ dev_info(dev,
|
||||
+ "Your board's MCU firmware does not support the LED gamma correction feature.\n");
|
||||
+ dev_info(dev,
|
||||
+ "Consider upgrading MCU firmware with the omnia-mcutool utility.\n");
|
||||
+ }
|
||||
+
|
||||
mutex_init(&leds->lock);
|
||||
|
||||
ret = devm_led_trigger_register(dev, &omnia_hw_trigger);
|
@ -0,0 +1,167 @@
|
||||
From ffec49d391c5f0195360912b216aa24dbc9b53c8 Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Marek=20Beh=C3=BAn?= <kabel@kernel.org>
|
||||
Date: Mon, 16 Oct 2023 16:15:38 +0200
|
||||
Subject: [PATCH] leds: turris-omnia: Fix brightness setting and trigger
|
||||
activating
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
I have improperly refactored commits
|
||||
4d5ed2621c24 ("leds: turris-omnia: Make set_brightness() more efficient")
|
||||
and
|
||||
aaf38273cf76 ("leds: turris-omnia: Support HW controlled mode via private trigger")
|
||||
after Lee requested a change in API semantics of the new functions I
|
||||
introduced in commit
|
||||
28350bc0ac77 ("leds: turris-omnia: Do not use SMBUS calls").
|
||||
|
||||
Before the change, the function omnia_cmd_write_u8() returned 0 on
|
||||
success, and afterwards it returned a positive value (number of bytes
|
||||
written). The latter version was applied, but the following commits did
|
||||
not properly account for this change.
|
||||
|
||||
This results in non-functional LED's .brightness_set_blocking() and
|
||||
trigger's .activate() methods.
|
||||
|
||||
The main reasoning behind the semantics change was that read/write
|
||||
methods should return the number of read/written bytes on success.
|
||||
It was pointed to me [1] that this is not always true (for example the
|
||||
regmap API does not do so), and since the driver never uses this number
|
||||
of read/written bytes information, I decided to fix this issue by
|
||||
changing the functions to the original semantics (return 0 on success).
|
||||
|
||||
[1] https://lore.kernel.org/linux-gpio/ZQnn+Gi0xVlsGCYA@smile.fi.intel.com/
|
||||
|
||||
Fixes: 28350bc0ac77 ("leds: turris-omnia: Do not use SMBUS calls")
|
||||
Signed-off-by: Marek Behún <kabel@kernel.org>
|
||||
---
|
||||
drivers/leds/leds-turris-omnia.c | 37 +++++++++++++++++---------------
|
||||
1 file changed, 20 insertions(+), 17 deletions(-)
|
||||
|
||||
--- a/drivers/leds/leds-turris-omnia.c
|
||||
+++ b/drivers/leds/leds-turris-omnia.c
|
||||
@@ -60,8 +60,11 @@ struct omnia_leds {
|
||||
static int omnia_cmd_write_u8(const struct i2c_client *client, u8 cmd, u8 val)
|
||||
{
|
||||
u8 buf[2] = { cmd, val };
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = i2c_master_send(client, buf, sizeof(buf));
|
||||
|
||||
- return i2c_master_send(client, buf, sizeof(buf));
|
||||
+ return ret < 0 ? ret : 0;
|
||||
}
|
||||
|
||||
static int omnia_cmd_read_raw(struct i2c_adapter *adapter, u8 addr, u8 cmd,
|
||||
@@ -81,7 +84,7 @@ static int omnia_cmd_read_raw(struct i2c
|
||||
|
||||
ret = i2c_transfer(adapter, msgs, ARRAY_SIZE(msgs));
|
||||
if (likely(ret == ARRAY_SIZE(msgs)))
|
||||
- return len;
|
||||
+ return 0;
|
||||
else if (ret < 0)
|
||||
return ret;
|
||||
else
|
||||
@@ -91,11 +94,11 @@ static int omnia_cmd_read_raw(struct i2c
|
||||
static int omnia_cmd_read_u8(const struct i2c_client *client, u8 cmd)
|
||||
{
|
||||
u8 reply;
|
||||
- int ret;
|
||||
+ int err;
|
||||
|
||||
- ret = omnia_cmd_read_raw(client->adapter, client->addr, cmd, &reply, 1);
|
||||
- if (ret < 0)
|
||||
- return ret;
|
||||
+ err = omnia_cmd_read_raw(client->adapter, client->addr, cmd, &reply, 1);
|
||||
+ if (err)
|
||||
+ return err;
|
||||
|
||||
return reply;
|
||||
}
|
||||
@@ -236,7 +239,7 @@ static void omnia_hwtrig_deactivate(stru
|
||||
|
||||
mutex_unlock(&leds->lock);
|
||||
|
||||
- if (err < 0)
|
||||
+ if (err)
|
||||
dev_err(cdev->dev, "Cannot put LED to software mode: %i\n",
|
||||
err);
|
||||
}
|
||||
@@ -302,7 +305,7 @@ static int omnia_led_register(struct i2c
|
||||
ret = omnia_cmd_write_u8(client, CMD_LED_MODE,
|
||||
CMD_LED_MODE_LED(led->reg) |
|
||||
CMD_LED_MODE_USER);
|
||||
- if (ret < 0) {
|
||||
+ if (ret) {
|
||||
dev_err(dev, "Cannot set LED %pOF to software mode: %i\n", np,
|
||||
ret);
|
||||
return ret;
|
||||
@@ -311,7 +314,7 @@ static int omnia_led_register(struct i2c
|
||||
/* disable the LED */
|
||||
ret = omnia_cmd_write_u8(client, CMD_LED_STATE,
|
||||
CMD_LED_STATE_LED(led->reg));
|
||||
- if (ret < 0) {
|
||||
+ if (ret) {
|
||||
dev_err(dev, "Cannot set LED %pOF brightness: %i\n", np, ret);
|
||||
return ret;
|
||||
}
|
||||
@@ -364,7 +367,7 @@ static ssize_t brightness_store(struct d
|
||||
{
|
||||
struct i2c_client *client = to_i2c_client(dev);
|
||||
unsigned long brightness;
|
||||
- int ret;
|
||||
+ int err;
|
||||
|
||||
if (kstrtoul(buf, 10, &brightness))
|
||||
return -EINVAL;
|
||||
@@ -372,9 +375,9 @@ static ssize_t brightness_store(struct d
|
||||
if (brightness > 100)
|
||||
return -EINVAL;
|
||||
|
||||
- ret = omnia_cmd_write_u8(client, CMD_LED_SET_BRIGHTNESS, brightness);
|
||||
+ err = omnia_cmd_write_u8(client, CMD_LED_SET_BRIGHTNESS, brightness);
|
||||
|
||||
- return ret < 0 ? ret : count;
|
||||
+ return err ?: count;
|
||||
}
|
||||
static DEVICE_ATTR_RW(brightness);
|
||||
|
||||
@@ -403,7 +406,7 @@ static ssize_t gamma_correction_store(st
|
||||
struct i2c_client *client = to_i2c_client(dev);
|
||||
struct omnia_leds *leds = i2c_get_clientdata(client);
|
||||
bool val;
|
||||
- int ret;
|
||||
+ int err;
|
||||
|
||||
if (!leds->has_gamma_correction)
|
||||
return -EOPNOTSUPP;
|
||||
@@ -411,9 +414,9 @@ static ssize_t gamma_correction_store(st
|
||||
if (kstrtobool(buf, &val) < 0)
|
||||
return -EINVAL;
|
||||
|
||||
- ret = omnia_cmd_write_u8(client, CMD_SET_GAMMA_CORRECTION, val);
|
||||
+ err = omnia_cmd_write_u8(client, CMD_SET_GAMMA_CORRECTION, val);
|
||||
|
||||
- return ret < 0 ? ret : count;
|
||||
+ return err ?: count;
|
||||
}
|
||||
static DEVICE_ATTR_RW(gamma_correction);
|
||||
|
||||
@@ -431,7 +434,7 @@ static int omnia_mcu_get_features(const
|
||||
|
||||
err = omnia_cmd_read_raw(client->adapter, OMNIA_MCU_I2C_ADDR,
|
||||
CMD_GET_STATUS_WORD, &reply, sizeof(reply));
|
||||
- if (err < 0)
|
||||
+ if (err)
|
||||
return err;
|
||||
|
||||
/* Check whether MCU firmware supports the CMD_GET_FEAUTRES command */
|
||||
@@ -440,7 +443,7 @@ static int omnia_mcu_get_features(const
|
||||
|
||||
err = omnia_cmd_read_raw(client->adapter, OMNIA_MCU_I2C_ADDR,
|
||||
CMD_GET_FEATURES, &reply, sizeof(reply));
|
||||
- if (err < 0)
|
||||
+ if (err)
|
||||
return err;
|
||||
|
||||
return le16_to_cpu(reply);
|
@ -0,0 +1,37 @@
|
||||
From 16724d6ea40a2c9315f5a0d81005dfa4d7a6da24 Mon Sep 17 00:00:00 2001
|
||||
From: Luca Weiss <luca.weiss@fairphone.com>
|
||||
Date: Fri, 20 Oct 2023 11:55:40 +0100
|
||||
Subject: [PATCH] nvmem: qfprom: Mark core clk as optional
|
||||
|
||||
On some platforms like sc7280 on non-ChromeOS devices the core clock
|
||||
cannot be touched by Linux so we cannot provide it. Mark it as optional
|
||||
as accessing qfprom for reading works without it but we still prohibit
|
||||
writing if we cannot provide the clock.
|
||||
|
||||
Signed-off-by: Luca Weiss <luca.weiss@fairphone.com>
|
||||
Reviewed-by: Douglas Anderson <dianders@chromium.org>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231020105545.216052-2-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/qfprom.c | 6 +++---
|
||||
1 file changed, 3 insertions(+), 3 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/qfprom.c
|
||||
+++ b/drivers/nvmem/qfprom.c
|
||||
@@ -423,12 +423,12 @@ static int qfprom_probe(struct platform_
|
||||
if (IS_ERR(priv->vcc))
|
||||
return PTR_ERR(priv->vcc);
|
||||
|
||||
- priv->secclk = devm_clk_get(dev, "core");
|
||||
+ priv->secclk = devm_clk_get_optional(dev, "core");
|
||||
if (IS_ERR(priv->secclk))
|
||||
return dev_err_probe(dev, PTR_ERR(priv->secclk), "Error getting clock\n");
|
||||
|
||||
- /* Only enable writing if we have SoC data. */
|
||||
- if (priv->soc_data)
|
||||
+ /* Only enable writing if we have SoC data and a valid clock */
|
||||
+ if (priv->soc_data && priv->secclk)
|
||||
econfig.reg_write = qfprom_reg_write;
|
||||
}
|
||||
|
@ -0,0 +1,330 @@
|
||||
From 2cc3b37f5b6df8189d55d0e812d9658ce256dfec Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
|
||||
Date: Fri, 20 Oct 2023 11:55:41 +0100
|
||||
Subject: [PATCH] nvmem: add explicit config option to read old syntax fixed OF
|
||||
cells
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
Binding for fixed NVMEM cells defined directly as NVMEM device subnodes
|
||||
has been deprecated. It has been replaced by the "fixed-layout" NVMEM
|
||||
layout binding.
|
||||
|
||||
New syntax is meant to be clearer and should help avoiding imprecise
|
||||
bindings.
|
||||
|
||||
NVMEM subsystem already supports the new binding. It should be a good
|
||||
idea to limit support for old syntax to existing drivers that actually
|
||||
support & use it (we can't break backward compatibility!). That way we
|
||||
additionally encourage new bindings & drivers to ignore deprecated
|
||||
binding.
|
||||
|
||||
It wasn't clear (to me) if rtc and w1 code actually uses old syntax
|
||||
fixed cells. I enabled them to don't risk any breakage.
|
||||
|
||||
Signed-off-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
[for meson-{efuse,mx-efuse}.c]
|
||||
Acked-by: Martin Blumenstingl <martin.blumenstingl@googlemail.com>
|
||||
[for mtk-efuse.c, nvmem/core.c, nvmem-provider.h]
|
||||
Reviewed-by: AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com>
|
||||
[MT8192, MT8195 Chromebooks]
|
||||
Tested-by: AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com>
|
||||
[for microchip-otpc.c]
|
||||
Reviewed-by: Claudiu Beznea <claudiu.beznea@microchip.com>
|
||||
[SAMA7G5-EK]
|
||||
Tested-by: Claudiu Beznea <claudiu.beznea@microchip.com>
|
||||
Acked-by: Jernej Skrabec <jernej.skrabec@gmail.com>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231020105545.216052-3-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/mtd/mtdcore.c | 2 ++
|
||||
drivers/nvmem/apple-efuses.c | 1 +
|
||||
drivers/nvmem/core.c | 8 +++++---
|
||||
drivers/nvmem/imx-ocotp-scu.c | 1 +
|
||||
drivers/nvmem/imx-ocotp.c | 1 +
|
||||
drivers/nvmem/meson-efuse.c | 1 +
|
||||
drivers/nvmem/meson-mx-efuse.c | 1 +
|
||||
drivers/nvmem/microchip-otpc.c | 1 +
|
||||
drivers/nvmem/mtk-efuse.c | 1 +
|
||||
drivers/nvmem/qcom-spmi-sdam.c | 1 +
|
||||
drivers/nvmem/qfprom.c | 1 +
|
||||
drivers/nvmem/rave-sp-eeprom.c | 1 +
|
||||
drivers/nvmem/rockchip-efuse.c | 1 +
|
||||
drivers/nvmem/sc27xx-efuse.c | 1 +
|
||||
drivers/nvmem/sec-qfprom.c | 1 +
|
||||
drivers/nvmem/sprd-efuse.c | 1 +
|
||||
drivers/nvmem/stm32-romem.c | 1 +
|
||||
drivers/nvmem/sunplus-ocotp.c | 1 +
|
||||
drivers/nvmem/sunxi_sid.c | 1 +
|
||||
drivers/nvmem/uniphier-efuse.c | 1 +
|
||||
drivers/nvmem/zynqmp_nvmem.c | 1 +
|
||||
drivers/rtc/nvmem.c | 1 +
|
||||
drivers/w1/slaves/w1_ds250x.c | 1 +
|
||||
include/linux/nvmem-provider.h | 2 ++
|
||||
24 files changed, 30 insertions(+), 3 deletions(-)
|
||||
|
||||
--- a/drivers/mtd/mtdcore.c
|
||||
+++ b/drivers/mtd/mtdcore.c
|
||||
@@ -552,6 +552,7 @@ static int mtd_nvmem_add(struct mtd_info
|
||||
config.dev = &mtd->dev;
|
||||
config.name = dev_name(&mtd->dev);
|
||||
config.owner = THIS_MODULE;
|
||||
+ config.add_legacy_fixed_of_cells = of_device_is_compatible(node, "nvmem-cells");
|
||||
config.reg_read = mtd_nvmem_reg_read;
|
||||
config.size = mtd->size;
|
||||
config.word_size = 1;
|
||||
@@ -898,6 +899,7 @@ static struct nvmem_device *mtd_otp_nvme
|
||||
config.name = compatible;
|
||||
config.id = NVMEM_DEVID_AUTO;
|
||||
config.owner = THIS_MODULE;
|
||||
+ config.add_legacy_fixed_of_cells = true;
|
||||
config.type = NVMEM_TYPE_OTP;
|
||||
config.root_only = true;
|
||||
config.ignore_wp = true;
|
||||
--- a/drivers/nvmem/apple-efuses.c
|
||||
+++ b/drivers/nvmem/apple-efuses.c
|
||||
@@ -36,6 +36,7 @@ static int apple_efuses_probe(struct pla
|
||||
struct resource *res;
|
||||
struct nvmem_config config = {
|
||||
.dev = &pdev->dev,
|
||||
+ .add_legacy_fixed_of_cells = true,
|
||||
.read_only = true,
|
||||
.reg_read = apple_efuses_read,
|
||||
.stride = sizeof(u32),
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -1003,9 +1003,11 @@ struct nvmem_device *nvmem_register(cons
|
||||
if (rval)
|
||||
goto err_remove_cells;
|
||||
|
||||
- rval = nvmem_add_cells_from_legacy_of(nvmem);
|
||||
- if (rval)
|
||||
- goto err_remove_cells;
|
||||
+ if (config->add_legacy_fixed_of_cells) {
|
||||
+ rval = nvmem_add_cells_from_legacy_of(nvmem);
|
||||
+ if (rval)
|
||||
+ goto err_remove_cells;
|
||||
+ }
|
||||
|
||||
rval = nvmem_add_cells_from_fixed_layout(nvmem);
|
||||
if (rval)
|
||||
--- a/drivers/nvmem/imx-ocotp-scu.c
|
||||
+++ b/drivers/nvmem/imx-ocotp-scu.c
|
||||
@@ -220,6 +220,7 @@ static int imx_scu_ocotp_write(void *con
|
||||
|
||||
static struct nvmem_config imx_scu_ocotp_nvmem_config = {
|
||||
.name = "imx-scu-ocotp",
|
||||
+ .add_legacy_fixed_of_cells = true,
|
||||
.read_only = false,
|
||||
.word_size = 4,
|
||||
.stride = 1,
|
||||
--- a/drivers/nvmem/imx-ocotp.c
|
||||
+++ b/drivers/nvmem/imx-ocotp.c
|
||||
@@ -615,6 +615,7 @@ static int imx_ocotp_probe(struct platfo
|
||||
return PTR_ERR(priv->clk);
|
||||
|
||||
priv->params = of_device_get_match_data(&pdev->dev);
|
||||
+ imx_ocotp_nvmem_config.add_legacy_fixed_of_cells = true;
|
||||
imx_ocotp_nvmem_config.size = 4 * priv->params->nregs;
|
||||
imx_ocotp_nvmem_config.dev = dev;
|
||||
imx_ocotp_nvmem_config.priv = priv;
|
||||
--- a/drivers/nvmem/meson-efuse.c
|
||||
+++ b/drivers/nvmem/meson-efuse.c
|
||||
@@ -93,6 +93,7 @@ static int meson_efuse_probe(struct plat
|
||||
|
||||
econfig->dev = dev;
|
||||
econfig->name = dev_name(dev);
|
||||
+ econfig->add_legacy_fixed_of_cells = true;
|
||||
econfig->stride = 1;
|
||||
econfig->word_size = 1;
|
||||
econfig->reg_read = meson_efuse_read;
|
||||
--- a/drivers/nvmem/meson-mx-efuse.c
|
||||
+++ b/drivers/nvmem/meson-mx-efuse.c
|
||||
@@ -210,6 +210,7 @@ static int meson_mx_efuse_probe(struct p
|
||||
efuse->config.owner = THIS_MODULE;
|
||||
efuse->config.dev = &pdev->dev;
|
||||
efuse->config.priv = efuse;
|
||||
+ efuse->config.add_legacy_fixed_of_cells = true;
|
||||
efuse->config.stride = drvdata->word_size;
|
||||
efuse->config.word_size = drvdata->word_size;
|
||||
efuse->config.size = SZ_512;
|
||||
--- a/drivers/nvmem/microchip-otpc.c
|
||||
+++ b/drivers/nvmem/microchip-otpc.c
|
||||
@@ -261,6 +261,7 @@ static int mchp_otpc_probe(struct platfo
|
||||
return ret;
|
||||
|
||||
mchp_nvmem_config.dev = otpc->dev;
|
||||
+ mchp_nvmem_config.add_legacy_fixed_of_cells = true;
|
||||
mchp_nvmem_config.size = size;
|
||||
mchp_nvmem_config.priv = otpc;
|
||||
nvmem = devm_nvmem_register(&pdev->dev, &mchp_nvmem_config);
|
||||
--- a/drivers/nvmem/mtk-efuse.c
|
||||
+++ b/drivers/nvmem/mtk-efuse.c
|
||||
@@ -83,6 +83,7 @@ static int mtk_efuse_probe(struct platfo
|
||||
return PTR_ERR(priv->base);
|
||||
|
||||
pdata = device_get_match_data(dev);
|
||||
+ econfig.add_legacy_fixed_of_cells = true;
|
||||
econfig.stride = 1;
|
||||
econfig.word_size = 1;
|
||||
econfig.reg_read = mtk_reg_read;
|
||||
--- a/drivers/nvmem/qcom-spmi-sdam.c
|
||||
+++ b/drivers/nvmem/qcom-spmi-sdam.c
|
||||
@@ -142,6 +142,7 @@ static int sdam_probe(struct platform_de
|
||||
sdam->sdam_config.name = "spmi_sdam";
|
||||
sdam->sdam_config.id = NVMEM_DEVID_AUTO;
|
||||
sdam->sdam_config.owner = THIS_MODULE;
|
||||
+ sdam->sdam_config.add_legacy_fixed_of_cells = true;
|
||||
sdam->sdam_config.stride = 1;
|
||||
sdam->sdam_config.word_size = 1;
|
||||
sdam->sdam_config.reg_read = sdam_read;
|
||||
--- a/drivers/nvmem/qfprom.c
|
||||
+++ b/drivers/nvmem/qfprom.c
|
||||
@@ -357,6 +357,7 @@ static int qfprom_probe(struct platform_
|
||||
{
|
||||
struct nvmem_config econfig = {
|
||||
.name = "qfprom",
|
||||
+ .add_legacy_fixed_of_cells = true,
|
||||
.stride = 1,
|
||||
.word_size = 1,
|
||||
.id = NVMEM_DEVID_AUTO,
|
||||
--- a/drivers/nvmem/rave-sp-eeprom.c
|
||||
+++ b/drivers/nvmem/rave-sp-eeprom.c
|
||||
@@ -328,6 +328,7 @@ static int rave_sp_eeprom_probe(struct p
|
||||
of_property_read_string(np, "zii,eeprom-name", &config.name);
|
||||
config.priv = eeprom;
|
||||
config.dev = dev;
|
||||
+ config.add_legacy_fixed_of_cells = true;
|
||||
config.size = size;
|
||||
config.reg_read = rave_sp_eeprom_reg_read;
|
||||
config.reg_write = rave_sp_eeprom_reg_write;
|
||||
--- a/drivers/nvmem/rockchip-efuse.c
|
||||
+++ b/drivers/nvmem/rockchip-efuse.c
|
||||
@@ -205,6 +205,7 @@ static int rockchip_rk3399_efuse_read(vo
|
||||
|
||||
static struct nvmem_config econfig = {
|
||||
.name = "rockchip-efuse",
|
||||
+ .add_legacy_fixed_of_cells = true,
|
||||
.stride = 1,
|
||||
.word_size = 1,
|
||||
.read_only = true,
|
||||
--- a/drivers/nvmem/sc27xx-efuse.c
|
||||
+++ b/drivers/nvmem/sc27xx-efuse.c
|
||||
@@ -247,6 +247,7 @@ static int sc27xx_efuse_probe(struct pla
|
||||
econfig.reg_read = sc27xx_efuse_read;
|
||||
econfig.priv = efuse;
|
||||
econfig.dev = &pdev->dev;
|
||||
+ econfig.add_legacy_fixed_of_cells = true;
|
||||
nvmem = devm_nvmem_register(&pdev->dev, &econfig);
|
||||
if (IS_ERR(nvmem)) {
|
||||
dev_err(&pdev->dev, "failed to register nvmem config\n");
|
||||
--- a/drivers/nvmem/sec-qfprom.c
|
||||
+++ b/drivers/nvmem/sec-qfprom.c
|
||||
@@ -47,6 +47,7 @@ static int sec_qfprom_probe(struct platf
|
||||
{
|
||||
struct nvmem_config econfig = {
|
||||
.name = "sec-qfprom",
|
||||
+ .add_legacy_fixed_of_cells = true,
|
||||
.stride = 1,
|
||||
.word_size = 1,
|
||||
.id = NVMEM_DEVID_AUTO,
|
||||
--- a/drivers/nvmem/sprd-efuse.c
|
||||
+++ b/drivers/nvmem/sprd-efuse.c
|
||||
@@ -408,6 +408,7 @@ static int sprd_efuse_probe(struct platf
|
||||
econfig.read_only = false;
|
||||
econfig.name = "sprd-efuse";
|
||||
econfig.size = efuse->data->blk_nums * SPRD_EFUSE_BLOCK_WIDTH;
|
||||
+ econfig.add_legacy_fixed_of_cells = true;
|
||||
econfig.reg_read = sprd_efuse_read;
|
||||
econfig.reg_write = sprd_efuse_write;
|
||||
econfig.priv = efuse;
|
||||
--- a/drivers/nvmem/stm32-romem.c
|
||||
+++ b/drivers/nvmem/stm32-romem.c
|
||||
@@ -207,6 +207,7 @@ static int stm32_romem_probe(struct plat
|
||||
priv->cfg.priv = priv;
|
||||
priv->cfg.owner = THIS_MODULE;
|
||||
priv->cfg.type = NVMEM_TYPE_OTP;
|
||||
+ priv->cfg.add_legacy_fixed_of_cells = true;
|
||||
|
||||
priv->lower = 0;
|
||||
|
||||
--- a/drivers/nvmem/sunplus-ocotp.c
|
||||
+++ b/drivers/nvmem/sunplus-ocotp.c
|
||||
@@ -145,6 +145,7 @@ disable_clk:
|
||||
|
||||
static struct nvmem_config sp_ocotp_nvmem_config = {
|
||||
.name = "sp-ocotp",
|
||||
+ .add_legacy_fixed_of_cells = true,
|
||||
.read_only = true,
|
||||
.word_size = 1,
|
||||
.size = QAC628_OTP_SIZE,
|
||||
--- a/drivers/nvmem/sunxi_sid.c
|
||||
+++ b/drivers/nvmem/sunxi_sid.c
|
||||
@@ -153,6 +153,7 @@ static int sunxi_sid_probe(struct platfo
|
||||
nvmem_cfg->dev = dev;
|
||||
nvmem_cfg->name = "sunxi-sid";
|
||||
nvmem_cfg->type = NVMEM_TYPE_OTP;
|
||||
+ nvmem_cfg->add_legacy_fixed_of_cells = true;
|
||||
nvmem_cfg->read_only = true;
|
||||
nvmem_cfg->size = cfg->size;
|
||||
nvmem_cfg->word_size = 1;
|
||||
--- a/drivers/nvmem/uniphier-efuse.c
|
||||
+++ b/drivers/nvmem/uniphier-efuse.c
|
||||
@@ -52,6 +52,7 @@ static int uniphier_efuse_probe(struct p
|
||||
econfig.size = resource_size(res);
|
||||
econfig.priv = priv;
|
||||
econfig.dev = dev;
|
||||
+ econfig.add_legacy_fixed_of_cells = true;
|
||||
nvmem = devm_nvmem_register(dev, &econfig);
|
||||
|
||||
return PTR_ERR_OR_ZERO(nvmem);
|
||||
--- a/drivers/nvmem/zynqmp_nvmem.c
|
||||
+++ b/drivers/nvmem/zynqmp_nvmem.c
|
||||
@@ -58,6 +58,7 @@ static int zynqmp_nvmem_probe(struct pla
|
||||
|
||||
priv->dev = dev;
|
||||
econfig.dev = dev;
|
||||
+ econfig.add_legacy_fixed_of_cells = true;
|
||||
econfig.reg_read = zynqmp_nvmem_read;
|
||||
econfig.priv = priv;
|
||||
|
||||
--- a/drivers/rtc/nvmem.c
|
||||
+++ b/drivers/rtc/nvmem.c
|
||||
@@ -21,6 +21,7 @@ int devm_rtc_nvmem_register(struct rtc_d
|
||||
|
||||
nvmem_config->dev = dev;
|
||||
nvmem_config->owner = rtc->owner;
|
||||
+ nvmem_config->add_legacy_fixed_of_cells = true;
|
||||
nvmem = devm_nvmem_register(dev, nvmem_config);
|
||||
if (IS_ERR(nvmem))
|
||||
dev_err(dev, "failed to register nvmem device for RTC\n");
|
||||
--- a/drivers/w1/slaves/w1_ds250x.c
|
||||
+++ b/drivers/w1/slaves/w1_ds250x.c
|
||||
@@ -168,6 +168,7 @@ static int w1_eprom_add_slave(struct w1_
|
||||
struct nvmem_device *nvmem;
|
||||
struct nvmem_config nvmem_cfg = {
|
||||
.dev = &sl->dev,
|
||||
+ .add_legacy_fixed_of_cells = true,
|
||||
.reg_read = w1_nvmem_read,
|
||||
.type = NVMEM_TYPE_OTP,
|
||||
.read_only = true,
|
||||
--- a/include/linux/nvmem-provider.h
|
||||
+++ b/include/linux/nvmem-provider.h
|
||||
@@ -82,6 +82,7 @@ struct nvmem_cell_info {
|
||||
* @owner: Pointer to exporter module. Used for refcounting.
|
||||
* @cells: Optional array of pre-defined NVMEM cells.
|
||||
* @ncells: Number of elements in cells.
|
||||
+ * @add_legacy_fixed_of_cells: Read fixed NVMEM cells from old OF syntax.
|
||||
* @keepout: Optional array of keepout ranges (sorted ascending by start).
|
||||
* @nkeepout: Number of elements in the keepout array.
|
||||
* @type: Type of the nvmem storage
|
||||
@@ -112,6 +113,7 @@ struct nvmem_config {
|
||||
struct module *owner;
|
||||
const struct nvmem_cell_info *cells;
|
||||
int ncells;
|
||||
+ bool add_legacy_fixed_of_cells;
|
||||
const struct nvmem_keepout *keepout;
|
||||
unsigned int nkeepout;
|
||||
enum nvmem_type type;
|
@ -0,0 +1,77 @@
|
||||
From 0720219f4d34a88a9badb4de70cfad7585687d48 Mon Sep 17 00:00:00 2001
|
||||
From: Rob Herring <robh@kernel.org>
|
||||
Date: Fri, 20 Oct 2023 11:55:45 +0100
|
||||
Subject: [PATCH] nvmem: Use device_get_match_data()
|
||||
|
||||
Use preferred device_get_match_data() instead of of_match_device() to
|
||||
get the driver match data. With this, adjust the includes to explicitly
|
||||
include the correct headers.
|
||||
|
||||
Signed-off-by: Rob Herring <robh@kernel.org>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231020105545.216052-7-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/mxs-ocotp.c | 10 ++++------
|
||||
drivers/nvmem/stm32-romem.c | 7 ++++---
|
||||
2 files changed, 8 insertions(+), 9 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/mxs-ocotp.c
|
||||
+++ b/drivers/nvmem/mxs-ocotp.c
|
||||
@@ -13,8 +13,9 @@
|
||||
#include <linux/io.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/nvmem-provider.h>
|
||||
-#include <linux/of_device.h>
|
||||
+#include <linux/of.h>
|
||||
#include <linux/platform_device.h>
|
||||
+#include <linux/property.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/stmp_device.h>
|
||||
|
||||
@@ -140,11 +141,10 @@ static int mxs_ocotp_probe(struct platfo
|
||||
struct device *dev = &pdev->dev;
|
||||
const struct mxs_data *data;
|
||||
struct mxs_ocotp *otp;
|
||||
- const struct of_device_id *match;
|
||||
int ret;
|
||||
|
||||
- match = of_match_device(dev->driver->of_match_table, dev);
|
||||
- if (!match || !match->data)
|
||||
+ data = device_get_match_data(dev);
|
||||
+ if (!data)
|
||||
return -EINVAL;
|
||||
|
||||
otp = devm_kzalloc(dev, sizeof(*otp), GFP_KERNEL);
|
||||
@@ -169,8 +169,6 @@ static int mxs_ocotp_probe(struct platfo
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
- data = match->data;
|
||||
-
|
||||
ocotp_config.size = data->size;
|
||||
ocotp_config.priv = otp;
|
||||
ocotp_config.dev = dev;
|
||||
--- a/drivers/nvmem/stm32-romem.c
|
||||
+++ b/drivers/nvmem/stm32-romem.c
|
||||
@@ -10,7 +10,9 @@
|
||||
#include <linux/io.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/nvmem-provider.h>
|
||||
-#include <linux/of_device.h>
|
||||
+#include <linux/of.h>
|
||||
+#include <linux/platform_device.h>
|
||||
+#include <linux/property.h>
|
||||
#include <linux/tee_drv.h>
|
||||
|
||||
#include "stm32-bsec-optee-ta.h"
|
||||
@@ -211,8 +213,7 @@ static int stm32_romem_probe(struct plat
|
||||
|
||||
priv->lower = 0;
|
||||
|
||||
- cfg = (const struct stm32_romem_cfg *)
|
||||
- of_match_device(dev->driver->of_match_table, dev)->data;
|
||||
+ cfg = device_get_match_data(dev);
|
||||
if (!cfg) {
|
||||
priv->cfg.read_only = true;
|
||||
priv->cfg.size = resource_size(res);
|
@ -0,0 +1,77 @@
|
||||
From f4cf4e5db331a5ce69e3f0b21d322cac0f4e4b5d Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
|
||||
Date: Mon, 23 Oct 2023 12:27:59 +0200
|
||||
Subject: [PATCH] Revert "nvmem: add new config option"
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
This reverts commit 517f14d9cf3533d5ab4fded195ab6f80a92e378f.
|
||||
|
||||
Config option "no_of_node" is no longer needed since adding a more
|
||||
explicit and targeted option "add_legacy_fixed_of_cells".
|
||||
|
||||
That "no_of_node" config option was needed *earlier* to help mtd's case.
|
||||
|
||||
DT nodes of MTD partitions (that are also NVMEM devices) may contain
|
||||
subnodes. Those SHOULD NOT be treated as NVMEM fixed cells.
|
||||
|
||||
To prevent NVMEM core code from parsing subnodes a "no_of_node" option
|
||||
was added (and set to true in mtd) to make for_each_child_of_node() in
|
||||
NVMEM a no-op. That was a bit hacky because it was messing with
|
||||
"of_node" pointer to achieve some side-effect.
|
||||
|
||||
With the introduction of "add_legacy_fixed_of_cells" config option
|
||||
things got more explicit. MTD subsystem simply tells NVMEM when to look
|
||||
for fixed cells and there is no need to hack "of_node" pointer anymore.
|
||||
|
||||
Signed-off-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Reviewed-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231023102759.31529-1-zajec5@gmail.com
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/mtd/mtdcore.c | 1 -
|
||||
drivers/nvmem/core.c | 2 +-
|
||||
include/linux/nvmem-provider.h | 2 --
|
||||
3 files changed, 1 insertion(+), 4 deletions(-)
|
||||
|
||||
--- a/drivers/mtd/mtdcore.c
|
||||
+++ b/drivers/mtd/mtdcore.c
|
||||
@@ -560,7 +560,6 @@ static int mtd_nvmem_add(struct mtd_info
|
||||
config.read_only = true;
|
||||
config.root_only = true;
|
||||
config.ignore_wp = true;
|
||||
- config.no_of_node = !of_device_is_compatible(node, "nvmem-cells");
|
||||
config.priv = mtd;
|
||||
|
||||
mtd->nvmem = nvmem_register(&config);
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -941,7 +941,7 @@ struct nvmem_device *nvmem_register(cons
|
||||
nvmem->nkeepout = config->nkeepout;
|
||||
if (config->of_node)
|
||||
nvmem->dev.of_node = config->of_node;
|
||||
- else if (!config->no_of_node)
|
||||
+ else
|
||||
nvmem->dev.of_node = config->dev->of_node;
|
||||
|
||||
switch (config->id) {
|
||||
--- a/include/linux/nvmem-provider.h
|
||||
+++ b/include/linux/nvmem-provider.h
|
||||
@@ -89,7 +89,6 @@ struct nvmem_cell_info {
|
||||
* @read_only: Device is read-only.
|
||||
* @root_only: Device is accessibly to root only.
|
||||
* @of_node: If given, this will be used instead of the parent's of_node.
|
||||
- * @no_of_node: Device should not use the parent's of_node even if it's !NULL.
|
||||
* @reg_read: Callback to read data.
|
||||
* @reg_write: Callback to write data.
|
||||
* @size: Device size.
|
||||
@@ -122,7 +121,6 @@ struct nvmem_config {
|
||||
bool ignore_wp;
|
||||
struct nvmem_layout *layout;
|
||||
struct device_node *of_node;
|
||||
- bool no_of_node;
|
||||
nvmem_reg_read_t reg_read;
|
||||
nvmem_reg_write_t reg_write;
|
||||
int size;
|
@ -0,0 +1,140 @@
|
||||
From 7f38b70042fcaa49219045bd1a9a2836e27a58ac Mon Sep 17 00:00:00 2001
|
||||
From: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Date: Fri, 15 Dec 2023 11:15:27 +0000
|
||||
Subject: [PATCH] of: device: Export of_device_make_bus_id()
|
||||
|
||||
This helper is really handy to create unique device names based on their
|
||||
device tree path, we may need it outside of the OF core (in the NVMEM
|
||||
subsystem) so let's export it. As this helper has nothing patform
|
||||
specific, let's move it to of/device.c instead of of/platform.c so we
|
||||
can add its prototype to of_device.h.
|
||||
|
||||
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Acked-by: Rob Herring <robh@kernel.org>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231215111536.316972-2-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/of/device.c | 41 +++++++++++++++++++++++++++++++++++++++
|
||||
drivers/of/platform.c | 40 --------------------------------------
|
||||
include/linux/of_device.h | 6 ++++++
|
||||
3 files changed, 47 insertions(+), 40 deletions(-)
|
||||
|
||||
--- a/drivers/of/device.c
|
||||
+++ b/drivers/of/device.c
|
||||
@@ -304,3 +304,44 @@ int of_device_uevent_modalias(const stru
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(of_device_uevent_modalias);
|
||||
+
|
||||
+/**
|
||||
+ * of_device_make_bus_id - Use the device node data to assign a unique name
|
||||
+ * @dev: pointer to device structure that is linked to a device tree node
|
||||
+ *
|
||||
+ * This routine will first try using the translated bus address to
|
||||
+ * derive a unique name. If it cannot, then it will prepend names from
|
||||
+ * parent nodes until a unique name can be derived.
|
||||
+ */
|
||||
+void of_device_make_bus_id(struct device *dev)
|
||||
+{
|
||||
+ struct device_node *node = dev->of_node;
|
||||
+ const __be32 *reg;
|
||||
+ u64 addr;
|
||||
+ u32 mask;
|
||||
+
|
||||
+ /* Construct the name, using parent nodes if necessary to ensure uniqueness */
|
||||
+ while (node->parent) {
|
||||
+ /*
|
||||
+ * If the address can be translated, then that is as much
|
||||
+ * uniqueness as we need. Make it the first component and return
|
||||
+ */
|
||||
+ reg = of_get_property(node, "reg", NULL);
|
||||
+ if (reg && (addr = of_translate_address(node, reg)) != OF_BAD_ADDR) {
|
||||
+ if (!of_property_read_u32(node, "mask", &mask))
|
||||
+ dev_set_name(dev, dev_name(dev) ? "%llx.%x.%pOFn:%s" : "%llx.%x.%pOFn",
|
||||
+ addr, ffs(mask) - 1, node, dev_name(dev));
|
||||
+
|
||||
+ else
|
||||
+ dev_set_name(dev, dev_name(dev) ? "%llx.%pOFn:%s" : "%llx.%pOFn",
|
||||
+ addr, node, dev_name(dev));
|
||||
+ return;
|
||||
+ }
|
||||
+
|
||||
+ /* format arguments only used if dev_name() resolves to NULL */
|
||||
+ dev_set_name(dev, dev_name(dev) ? "%s:%s" : "%s",
|
||||
+ kbasename(node->full_name), dev_name(dev));
|
||||
+ node = node->parent;
|
||||
+ }
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(of_device_make_bus_id);
|
||||
--- a/drivers/of/platform.c
|
||||
+++ b/drivers/of/platform.c
|
||||
@@ -98,46 +98,6 @@ static const struct of_device_id of_skip
|
||||
*/
|
||||
|
||||
/**
|
||||
- * of_device_make_bus_id - Use the device node data to assign a unique name
|
||||
- * @dev: pointer to device structure that is linked to a device tree node
|
||||
- *
|
||||
- * This routine will first try using the translated bus address to
|
||||
- * derive a unique name. If it cannot, then it will prepend names from
|
||||
- * parent nodes until a unique name can be derived.
|
||||
- */
|
||||
-static void of_device_make_bus_id(struct device *dev)
|
||||
-{
|
||||
- struct device_node *node = dev->of_node;
|
||||
- const __be32 *reg;
|
||||
- u64 addr;
|
||||
- u32 mask;
|
||||
-
|
||||
- /* Construct the name, using parent nodes if necessary to ensure uniqueness */
|
||||
- while (node->parent) {
|
||||
- /*
|
||||
- * If the address can be translated, then that is as much
|
||||
- * uniqueness as we need. Make it the first component and return
|
||||
- */
|
||||
- reg = of_get_property(node, "reg", NULL);
|
||||
- if (reg && (addr = of_translate_address(node, reg)) != OF_BAD_ADDR) {
|
||||
- if (!of_property_read_u32(node, "mask", &mask))
|
||||
- dev_set_name(dev, dev_name(dev) ? "%llx.%x.%pOFn:%s" : "%llx.%x.%pOFn",
|
||||
- addr, ffs(mask) - 1, node, dev_name(dev));
|
||||
-
|
||||
- else
|
||||
- dev_set_name(dev, dev_name(dev) ? "%llx.%pOFn:%s" : "%llx.%pOFn",
|
||||
- addr, node, dev_name(dev));
|
||||
- return;
|
||||
- }
|
||||
-
|
||||
- /* format arguments only used if dev_name() resolves to NULL */
|
||||
- dev_set_name(dev, dev_name(dev) ? "%s:%s" : "%s",
|
||||
- kbasename(node->full_name), dev_name(dev));
|
||||
- node = node->parent;
|
||||
- }
|
||||
-}
|
||||
-
|
||||
-/**
|
||||
* of_device_alloc - Allocate and initialize an of_device
|
||||
* @np: device node to assign to device
|
||||
* @bus_id: Name to assign to the device. May be null to use default name.
|
||||
--- a/include/linux/of_device.h
|
||||
+++ b/include/linux/of_device.h
|
||||
@@ -40,6 +40,9 @@ static inline int of_dma_configure(struc
|
||||
{
|
||||
return of_dma_configure_id(dev, np, force_dma, NULL);
|
||||
}
|
||||
+
|
||||
+void of_device_make_bus_id(struct device *dev);
|
||||
+
|
||||
#else /* CONFIG_OF */
|
||||
|
||||
static inline int of_driver_match_device(struct device *dev,
|
||||
@@ -82,6 +85,9 @@ static inline int of_dma_configure(struc
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
+
|
||||
+static inline void of_device_make_bus_id(struct device *dev) {}
|
||||
+
|
||||
#endif /* CONFIG_OF */
|
||||
|
||||
#endif /* _LINUX_OF_DEVICE_H */
|
@ -0,0 +1,95 @@
|
||||
From 4a1a40233b4a9fc159a5c7a27dc34c5c7bc5be55 Mon Sep 17 00:00:00 2001
|
||||
From: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Date: Fri, 15 Dec 2023 11:15:28 +0000
|
||||
Subject: [PATCH] nvmem: Move of_nvmem_layout_get_container() in another header
|
||||
|
||||
nvmem-consumer.h is included by consumer devices, extracting data from
|
||||
NVMEM devices whereas nvmem-provider.h is included by devices providing
|
||||
NVMEM content.
|
||||
|
||||
The only users of of_nvmem_layout_get_container() outside of the core
|
||||
are layout drivers, so better move its prototype to nvmem-provider.h.
|
||||
|
||||
While we do so, we also move the kdoc associated with the function to
|
||||
the header rather than the .c file.
|
||||
|
||||
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231215111536.316972-3-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/core.c | 8 --------
|
||||
include/linux/nvmem-consumer.h | 7 -------
|
||||
include/linux/nvmem-provider.h | 21 +++++++++++++++++++++
|
||||
3 files changed, 21 insertions(+), 15 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -847,14 +847,6 @@ static int nvmem_add_cells_from_layout(s
|
||||
}
|
||||
|
||||
#if IS_ENABLED(CONFIG_OF)
|
||||
-/**
|
||||
- * of_nvmem_layout_get_container() - Get OF node to layout container.
|
||||
- *
|
||||
- * @nvmem: nvmem device.
|
||||
- *
|
||||
- * Return: a node pointer with refcount incremented or NULL if no
|
||||
- * container exists. Use of_node_put() on it when done.
|
||||
- */
|
||||
struct device_node *of_nvmem_layout_get_container(struct nvmem_device *nvmem)
|
||||
{
|
||||
return of_get_child_by_name(nvmem->dev.of_node, "nvmem-layout");
|
||||
--- a/include/linux/nvmem-consumer.h
|
||||
+++ b/include/linux/nvmem-consumer.h
|
||||
@@ -241,7 +241,6 @@ struct nvmem_cell *of_nvmem_cell_get(str
|
||||
const char *id);
|
||||
struct nvmem_device *of_nvmem_device_get(struct device_node *np,
|
||||
const char *name);
|
||||
-struct device_node *of_nvmem_layout_get_container(struct nvmem_device *nvmem);
|
||||
#else
|
||||
static inline struct nvmem_cell *of_nvmem_cell_get(struct device_node *np,
|
||||
const char *id)
|
||||
@@ -254,12 +253,6 @@ static inline struct nvmem_device *of_nv
|
||||
{
|
||||
return ERR_PTR(-EOPNOTSUPP);
|
||||
}
|
||||
-
|
||||
-static inline struct device_node *
|
||||
-of_nvmem_layout_get_container(struct nvmem_device *nvmem)
|
||||
-{
|
||||
- return NULL;
|
||||
-}
|
||||
#endif /* CONFIG_NVMEM && CONFIG_OF */
|
||||
|
||||
#endif /* ifndef _LINUX_NVMEM_CONSUMER_H */
|
||||
--- a/include/linux/nvmem-provider.h
|
||||
+++ b/include/linux/nvmem-provider.h
|
||||
@@ -244,6 +244,27 @@ nvmem_layout_get_match_data(struct nvmem
|
||||
|
||||
#endif /* CONFIG_NVMEM */
|
||||
|
||||
+#if IS_ENABLED(CONFIG_NVMEM) && IS_ENABLED(CONFIG_OF)
|
||||
+
|
||||
+/**
|
||||
+ * of_nvmem_layout_get_container() - Get OF node of layout container
|
||||
+ *
|
||||
+ * @nvmem: nvmem device
|
||||
+ *
|
||||
+ * Return: a node pointer with refcount incremented or NULL if no
|
||||
+ * container exists. Use of_node_put() on it when done.
|
||||
+ */
|
||||
+struct device_node *of_nvmem_layout_get_container(struct nvmem_device *nvmem);
|
||||
+
|
||||
+#else /* CONFIG_NVMEM && CONFIG_OF */
|
||||
+
|
||||
+static inline struct device_node *of_nvmem_layout_get_container(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ return NULL;
|
||||
+}
|
||||
+
|
||||
+#endif /* CONFIG_NVMEM && CONFIG_OF */
|
||||
+
|
||||
#define module_nvmem_layout_driver(__layout_driver) \
|
||||
module_driver(__layout_driver, nvmem_layout_register, \
|
||||
nvmem_layout_unregister)
|
@ -0,0 +1,91 @@
|
||||
From ec9c08a1cb8dc5e8e003f95f5f62de41dde235bb Mon Sep 17 00:00:00 2001
|
||||
From: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Date: Fri, 15 Dec 2023 11:15:29 +0000
|
||||
Subject: [PATCH] nvmem: Create a header for internal sharing
|
||||
|
||||
Before adding all the NVMEM layout bus infrastructure to the core, let's
|
||||
move the main nvmem_device structure in an internal header, only
|
||||
available to the core. This way all the additional code can be added in
|
||||
a dedicated file in order to keep the current core file tidy.
|
||||
|
||||
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231215111536.316972-4-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/core.c | 24 +-----------------------
|
||||
drivers/nvmem/internals.h | 35 +++++++++++++++++++++++++++++++++++
|
||||
2 files changed, 36 insertions(+), 23 deletions(-)
|
||||
create mode 100644 drivers/nvmem/internals.h
|
||||
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -19,29 +19,7 @@
|
||||
#include <linux/of.h>
|
||||
#include <linux/slab.h>
|
||||
|
||||
-struct nvmem_device {
|
||||
- struct module *owner;
|
||||
- struct device dev;
|
||||
- int stride;
|
||||
- int word_size;
|
||||
- int id;
|
||||
- struct kref refcnt;
|
||||
- size_t size;
|
||||
- bool read_only;
|
||||
- bool root_only;
|
||||
- int flags;
|
||||
- enum nvmem_type type;
|
||||
- struct bin_attribute eeprom;
|
||||
- struct device *base_dev;
|
||||
- struct list_head cells;
|
||||
- const struct nvmem_keepout *keepout;
|
||||
- unsigned int nkeepout;
|
||||
- nvmem_reg_read_t reg_read;
|
||||
- nvmem_reg_write_t reg_write;
|
||||
- struct gpio_desc *wp_gpio;
|
||||
- struct nvmem_layout *layout;
|
||||
- void *priv;
|
||||
-};
|
||||
+#include "internals.h"
|
||||
|
||||
#define to_nvmem_device(d) container_of(d, struct nvmem_device, dev)
|
||||
|
||||
--- /dev/null
|
||||
+++ b/drivers/nvmem/internals.h
|
||||
@@ -0,0 +1,35 @@
|
||||
+/* SPDX-License-Identifier: GPL-2.0 */
|
||||
+
|
||||
+#ifndef _LINUX_NVMEM_INTERNALS_H
|
||||
+#define _LINUX_NVMEM_INTERNALS_H
|
||||
+
|
||||
+#include <linux/device.h>
|
||||
+#include <linux/nvmem-consumer.h>
|
||||
+#include <linux/nvmem-provider.h>
|
||||
+
|
||||
+struct nvmem_device {
|
||||
+ struct module *owner;
|
||||
+ struct device dev;
|
||||
+ struct list_head node;
|
||||
+ int stride;
|
||||
+ int word_size;
|
||||
+ int id;
|
||||
+ struct kref refcnt;
|
||||
+ size_t size;
|
||||
+ bool read_only;
|
||||
+ bool root_only;
|
||||
+ int flags;
|
||||
+ enum nvmem_type type;
|
||||
+ struct bin_attribute eeprom;
|
||||
+ struct device *base_dev;
|
||||
+ struct list_head cells;
|
||||
+ const struct nvmem_keepout *keepout;
|
||||
+ unsigned int nkeepout;
|
||||
+ nvmem_reg_read_t reg_read;
|
||||
+ nvmem_reg_write_t reg_write;
|
||||
+ struct gpio_desc *wp_gpio;
|
||||
+ struct nvmem_layout *layout;
|
||||
+ void *priv;
|
||||
+};
|
||||
+
|
||||
+#endif /* ifndef _LINUX_NVMEM_INTERNALS_H */
|
@ -0,0 +1,79 @@
|
||||
From 1b7c298a4ecbc28cc6ee94005734bff55eb83d22 Mon Sep 17 00:00:00 2001
|
||||
From: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Date: Fri, 15 Dec 2023 11:15:30 +0000
|
||||
Subject: [PATCH] nvmem: Simplify the ->add_cells() hook
|
||||
|
||||
The layout entry is not used and will anyway be made useless by the new
|
||||
layout bus infrastructure coming next, so drop it. While at it, clarify
|
||||
the kdoc entry.
|
||||
|
||||
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231215111536.316972-5-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/core.c | 2 +-
|
||||
drivers/nvmem/layouts/onie-tlv.c | 3 +--
|
||||
drivers/nvmem/layouts/sl28vpd.c | 3 +--
|
||||
include/linux/nvmem-provider.h | 8 +++-----
|
||||
4 files changed, 6 insertions(+), 10 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -816,7 +816,7 @@ static int nvmem_add_cells_from_layout(s
|
||||
int ret;
|
||||
|
||||
if (layout && layout->add_cells) {
|
||||
- ret = layout->add_cells(&nvmem->dev, nvmem, layout);
|
||||
+ ret = layout->add_cells(&nvmem->dev, nvmem);
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
--- a/drivers/nvmem/layouts/onie-tlv.c
|
||||
+++ b/drivers/nvmem/layouts/onie-tlv.c
|
||||
@@ -182,8 +182,7 @@ static bool onie_tlv_crc_is_valid(struct
|
||||
return true;
|
||||
}
|
||||
|
||||
-static int onie_tlv_parse_table(struct device *dev, struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout)
|
||||
+static int onie_tlv_parse_table(struct device *dev, struct nvmem_device *nvmem)
|
||||
{
|
||||
struct onie_tlv_hdr hdr;
|
||||
size_t table_len, data_len, hdr_len;
|
||||
--- a/drivers/nvmem/layouts/sl28vpd.c
|
||||
+++ b/drivers/nvmem/layouts/sl28vpd.c
|
||||
@@ -80,8 +80,7 @@ static int sl28vpd_v1_check_crc(struct d
|
||||
return 0;
|
||||
}
|
||||
|
||||
-static int sl28vpd_add_cells(struct device *dev, struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout)
|
||||
+static int sl28vpd_add_cells(struct device *dev, struct nvmem_device *nvmem)
|
||||
{
|
||||
const struct nvmem_cell_info *pinfo;
|
||||
struct nvmem_cell_info info = {0};
|
||||
--- a/include/linux/nvmem-provider.h
|
||||
+++ b/include/linux/nvmem-provider.h
|
||||
@@ -156,9 +156,8 @@ struct nvmem_cell_table {
|
||||
*
|
||||
* @name: Layout name.
|
||||
* @of_match_table: Open firmware match table.
|
||||
- * @add_cells: Will be called if a nvmem device is found which
|
||||
- * has this layout. The function will add layout
|
||||
- * specific cells with nvmem_add_one_cell().
|
||||
+ * @add_cells: Called to populate the layout using
|
||||
+ * nvmem_add_one_cell().
|
||||
* @fixup_cell_info: Will be called before a cell is added. Can be
|
||||
* used to modify the nvmem_cell_info.
|
||||
* @owner: Pointer to struct module.
|
||||
@@ -172,8 +171,7 @@ struct nvmem_cell_table {
|
||||
struct nvmem_layout {
|
||||
const char *name;
|
||||
const struct of_device_id *of_match_table;
|
||||
- int (*add_cells)(struct device *dev, struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout);
|
||||
+ int (*add_cells)(struct device *dev, struct nvmem_device *nvmem);
|
||||
void (*fixup_cell_info)(struct nvmem_device *nvmem,
|
||||
struct nvmem_layout *layout,
|
||||
struct nvmem_cell_info *cell);
|
@ -0,0 +1,169 @@
|
||||
From 1172460e716784ac7e1049a537bdca8edbf97360 Mon Sep 17 00:00:00 2001
|
||||
From: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Date: Fri, 15 Dec 2023 11:15:31 +0000
|
||||
Subject: [PATCH] nvmem: Move and rename ->fixup_cell_info()
|
||||
|
||||
This hook is meant to be used by any provider and instantiating a layout
|
||||
just for this is useless. Let's instead move this hook to the nvmem
|
||||
device and add it to the config structure to be easily shared by the
|
||||
providers.
|
||||
|
||||
While at moving this hook, rename it ->fixup_dt_cell_info() to clarify
|
||||
its main intended purpose.
|
||||
|
||||
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231215111536.316972-6-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/core.c | 6 +++---
|
||||
drivers/nvmem/imx-ocotp.c | 11 +++--------
|
||||
drivers/nvmem/internals.h | 2 ++
|
||||
drivers/nvmem/mtk-efuse.c | 11 +++--------
|
||||
include/linux/nvmem-provider.h | 9 ++++-----
|
||||
5 files changed, 15 insertions(+), 24 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -675,7 +675,6 @@ static int nvmem_validate_keepouts(struc
|
||||
|
||||
static int nvmem_add_cells_from_dt(struct nvmem_device *nvmem, struct device_node *np)
|
||||
{
|
||||
- struct nvmem_layout *layout = nvmem->layout;
|
||||
struct device *dev = &nvmem->dev;
|
||||
struct device_node *child;
|
||||
const __be32 *addr;
|
||||
@@ -705,8 +704,8 @@ static int nvmem_add_cells_from_dt(struc
|
||||
|
||||
info.np = of_node_get(child);
|
||||
|
||||
- if (layout && layout->fixup_cell_info)
|
||||
- layout->fixup_cell_info(nvmem, layout, &info);
|
||||
+ if (nvmem->fixup_dt_cell_info)
|
||||
+ nvmem->fixup_dt_cell_info(nvmem, &info);
|
||||
|
||||
ret = nvmem_add_one_cell(nvmem, &info);
|
||||
kfree(info.name);
|
||||
@@ -895,6 +894,7 @@ struct nvmem_device *nvmem_register(cons
|
||||
|
||||
kref_init(&nvmem->refcnt);
|
||||
INIT_LIST_HEAD(&nvmem->cells);
|
||||
+ nvmem->fixup_dt_cell_info = config->fixup_dt_cell_info;
|
||||
|
||||
nvmem->owner = config->owner;
|
||||
if (!nvmem->owner && config->dev->driver)
|
||||
--- a/drivers/nvmem/imx-ocotp.c
|
||||
+++ b/drivers/nvmem/imx-ocotp.c
|
||||
@@ -583,17 +583,12 @@ static const struct of_device_id imx_oco
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, imx_ocotp_dt_ids);
|
||||
|
||||
-static void imx_ocotp_fixup_cell_info(struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout,
|
||||
- struct nvmem_cell_info *cell)
|
||||
+static void imx_ocotp_fixup_dt_cell_info(struct nvmem_device *nvmem,
|
||||
+ struct nvmem_cell_info *cell)
|
||||
{
|
||||
cell->read_post_process = imx_ocotp_cell_pp;
|
||||
}
|
||||
|
||||
-static struct nvmem_layout imx_ocotp_layout = {
|
||||
- .fixup_cell_info = imx_ocotp_fixup_cell_info,
|
||||
-};
|
||||
-
|
||||
static int imx_ocotp_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct device *dev = &pdev->dev;
|
||||
@@ -619,7 +614,7 @@ static int imx_ocotp_probe(struct platfo
|
||||
imx_ocotp_nvmem_config.size = 4 * priv->params->nregs;
|
||||
imx_ocotp_nvmem_config.dev = dev;
|
||||
imx_ocotp_nvmem_config.priv = priv;
|
||||
- imx_ocotp_nvmem_config.layout = &imx_ocotp_layout;
|
||||
+ imx_ocotp_nvmem_config.fixup_dt_cell_info = &imx_ocotp_fixup_dt_cell_info;
|
||||
|
||||
priv->config = &imx_ocotp_nvmem_config;
|
||||
|
||||
--- a/drivers/nvmem/internals.h
|
||||
+++ b/drivers/nvmem/internals.h
|
||||
@@ -23,6 +23,8 @@ struct nvmem_device {
|
||||
struct bin_attribute eeprom;
|
||||
struct device *base_dev;
|
||||
struct list_head cells;
|
||||
+ void (*fixup_dt_cell_info)(struct nvmem_device *nvmem,
|
||||
+ struct nvmem_cell_info *cell);
|
||||
const struct nvmem_keepout *keepout;
|
||||
unsigned int nkeepout;
|
||||
nvmem_reg_read_t reg_read;
|
||||
--- a/drivers/nvmem/mtk-efuse.c
|
||||
+++ b/drivers/nvmem/mtk-efuse.c
|
||||
@@ -45,9 +45,8 @@ static int mtk_efuse_gpu_speedbin_pp(voi
|
||||
return 0;
|
||||
}
|
||||
|
||||
-static void mtk_efuse_fixup_cell_info(struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout,
|
||||
- struct nvmem_cell_info *cell)
|
||||
+static void mtk_efuse_fixup_dt_cell_info(struct nvmem_device *nvmem,
|
||||
+ struct nvmem_cell_info *cell)
|
||||
{
|
||||
size_t sz = strlen(cell->name);
|
||||
|
||||
@@ -61,10 +60,6 @@ static void mtk_efuse_fixup_cell_info(st
|
||||
cell->read_post_process = mtk_efuse_gpu_speedbin_pp;
|
||||
}
|
||||
|
||||
-static struct nvmem_layout mtk_efuse_layout = {
|
||||
- .fixup_cell_info = mtk_efuse_fixup_cell_info,
|
||||
-};
|
||||
-
|
||||
static int mtk_efuse_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct device *dev = &pdev->dev;
|
||||
@@ -91,7 +86,7 @@ static int mtk_efuse_probe(struct platfo
|
||||
econfig.priv = priv;
|
||||
econfig.dev = dev;
|
||||
if (pdata->uses_post_processing)
|
||||
- econfig.layout = &mtk_efuse_layout;
|
||||
+ econfig.fixup_dt_cell_info = &mtk_efuse_fixup_dt_cell_info;
|
||||
nvmem = devm_nvmem_register(dev, &econfig);
|
||||
|
||||
return PTR_ERR_OR_ZERO(nvmem);
|
||||
--- a/include/linux/nvmem-provider.h
|
||||
+++ b/include/linux/nvmem-provider.h
|
||||
@@ -83,6 +83,8 @@ struct nvmem_cell_info {
|
||||
* @cells: Optional array of pre-defined NVMEM cells.
|
||||
* @ncells: Number of elements in cells.
|
||||
* @add_legacy_fixed_of_cells: Read fixed NVMEM cells from old OF syntax.
|
||||
+ * @fixup_dt_cell_info: Will be called before a cell is added. Can be
|
||||
+ * used to modify the nvmem_cell_info.
|
||||
* @keepout: Optional array of keepout ranges (sorted ascending by start).
|
||||
* @nkeepout: Number of elements in the keepout array.
|
||||
* @type: Type of the nvmem storage
|
||||
@@ -113,6 +115,8 @@ struct nvmem_config {
|
||||
const struct nvmem_cell_info *cells;
|
||||
int ncells;
|
||||
bool add_legacy_fixed_of_cells;
|
||||
+ void (*fixup_dt_cell_info)(struct nvmem_device *nvmem,
|
||||
+ struct nvmem_cell_info *cell);
|
||||
const struct nvmem_keepout *keepout;
|
||||
unsigned int nkeepout;
|
||||
enum nvmem_type type;
|
||||
@@ -158,8 +162,6 @@ struct nvmem_cell_table {
|
||||
* @of_match_table: Open firmware match table.
|
||||
* @add_cells: Called to populate the layout using
|
||||
* nvmem_add_one_cell().
|
||||
- * @fixup_cell_info: Will be called before a cell is added. Can be
|
||||
- * used to modify the nvmem_cell_info.
|
||||
* @owner: Pointer to struct module.
|
||||
* @node: List node.
|
||||
*
|
||||
@@ -172,9 +174,6 @@ struct nvmem_layout {
|
||||
const char *name;
|
||||
const struct of_device_id *of_match_table;
|
||||
int (*add_cells)(struct device *dev, struct nvmem_device *nvmem);
|
||||
- void (*fixup_cell_info)(struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout,
|
||||
- struct nvmem_cell_info *cell);
|
||||
|
||||
/* private */
|
||||
struct module *owner;
|
@ -0,0 +1,763 @@
|
||||
From fc29fd821d9ac2ae3d32a722fac39ce874efb883 Mon Sep 17 00:00:00 2001
|
||||
From: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Date: Fri, 15 Dec 2023 11:15:32 +0000
|
||||
Subject: [PATCH] nvmem: core: Rework layouts to become regular devices
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
Current layout support was initially written without modules support in
|
||||
mind. When the requirement for module support rose, the existing base
|
||||
was improved to adopt modularization support, but kind of a design flaw
|
||||
was introduced. With the existing implementation, when a storage device
|
||||
registers into NVMEM, the core tries to hook a layout (if any) and
|
||||
populates its cells immediately. This means, if the hardware description
|
||||
expects a layout to be hooked up, but no driver was provided for that,
|
||||
the storage medium will fail to probe and try later from
|
||||
scratch. Even if we consider that the hardware description shall be
|
||||
correct, we could still probe the storage device (especially if it
|
||||
contains the rootfs).
|
||||
|
||||
One way to overcome this situation is to consider the layouts as
|
||||
devices, and leverage the native notifier mechanism. When a new NVMEM
|
||||
device is registered, we can populate its nvmem-layout child, if any,
|
||||
and wait for the matching to be done in order to get the cells (the
|
||||
waiting can be easily done with the NVMEM notifiers). If the layout
|
||||
driver is compiled as a module, it should automatically be loaded. This
|
||||
way, there is no strong order to enforce, any NVMEM device creation
|
||||
or NVMEM layout driver insertion will be observed as a new event which
|
||||
may lead to the creation of additional cells, without disturbing the
|
||||
probes with costly (and sometimes endless) deferrals.
|
||||
|
||||
In order to achieve that goal we create a new bus for the nvmem-layouts
|
||||
with minimal logic to match nvmem-layout devices with nvmem-layout
|
||||
drivers. All this infrastructure code is created in the layouts.c file.
|
||||
|
||||
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Tested-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231215111536.316972-7-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/Kconfig | 1 +
|
||||
drivers/nvmem/Makefile | 2 +
|
||||
drivers/nvmem/core.c | 170 ++++++++++----------------
|
||||
drivers/nvmem/internals.h | 21 ++++
|
||||
drivers/nvmem/layouts.c | 201 +++++++++++++++++++++++++++++++
|
||||
drivers/nvmem/layouts/Kconfig | 8 ++
|
||||
drivers/nvmem/layouts/onie-tlv.c | 24 +++-
|
||||
drivers/nvmem/layouts/sl28vpd.c | 24 +++-
|
||||
include/linux/nvmem-provider.h | 38 +++---
|
||||
9 files changed, 354 insertions(+), 135 deletions(-)
|
||||
create mode 100644 drivers/nvmem/layouts.c
|
||||
|
||||
--- a/drivers/nvmem/Kconfig
|
||||
+++ b/drivers/nvmem/Kconfig
|
||||
@@ -1,6 +1,7 @@
|
||||
# SPDX-License-Identifier: GPL-2.0-only
|
||||
menuconfig NVMEM
|
||||
bool "NVMEM Support"
|
||||
+ imply NVMEM_LAYOUTS
|
||||
help
|
||||
Support for NVMEM(Non Volatile Memory) devices like EEPROM, EFUSES...
|
||||
|
||||
--- a/drivers/nvmem/Makefile
|
||||
+++ b/drivers/nvmem/Makefile
|
||||
@@ -5,6 +5,8 @@
|
||||
|
||||
obj-$(CONFIG_NVMEM) += nvmem_core.o
|
||||
nvmem_core-y := core.o
|
||||
+obj-$(CONFIG_NVMEM_LAYOUTS) += nvmem_layouts.o
|
||||
+nvmem_layouts-y := layouts.o
|
||||
obj-y += layouts/
|
||||
|
||||
# Devices
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -55,9 +55,6 @@ static LIST_HEAD(nvmem_lookup_list);
|
||||
|
||||
static BLOCKING_NOTIFIER_HEAD(nvmem_notifier);
|
||||
|
||||
-static DEFINE_SPINLOCK(nvmem_layout_lock);
|
||||
-static LIST_HEAD(nvmem_layouts);
|
||||
-
|
||||
static int __nvmem_reg_read(struct nvmem_device *nvmem, unsigned int offset,
|
||||
void *val, size_t bytes)
|
||||
{
|
||||
@@ -740,97 +737,22 @@ static int nvmem_add_cells_from_fixed_la
|
||||
return err;
|
||||
}
|
||||
|
||||
-int __nvmem_layout_register(struct nvmem_layout *layout, struct module *owner)
|
||||
+int nvmem_layout_register(struct nvmem_layout *layout)
|
||||
{
|
||||
- layout->owner = owner;
|
||||
-
|
||||
- spin_lock(&nvmem_layout_lock);
|
||||
- list_add(&layout->node, &nvmem_layouts);
|
||||
- spin_unlock(&nvmem_layout_lock);
|
||||
-
|
||||
- blocking_notifier_call_chain(&nvmem_notifier, NVMEM_LAYOUT_ADD, layout);
|
||||
+ if (!layout->add_cells)
|
||||
+ return -EINVAL;
|
||||
|
||||
- return 0;
|
||||
+ /* Populate the cells */
|
||||
+ return layout->add_cells(&layout->nvmem->dev, layout->nvmem);
|
||||
}
|
||||
-EXPORT_SYMBOL_GPL(__nvmem_layout_register);
|
||||
+EXPORT_SYMBOL_GPL(nvmem_layout_register);
|
||||
|
||||
void nvmem_layout_unregister(struct nvmem_layout *layout)
|
||||
{
|
||||
- blocking_notifier_call_chain(&nvmem_notifier, NVMEM_LAYOUT_REMOVE, layout);
|
||||
-
|
||||
- spin_lock(&nvmem_layout_lock);
|
||||
- list_del(&layout->node);
|
||||
- spin_unlock(&nvmem_layout_lock);
|
||||
+ /* Keep the API even with an empty stub in case we need it later */
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(nvmem_layout_unregister);
|
||||
|
||||
-static struct nvmem_layout *nvmem_layout_get(struct nvmem_device *nvmem)
|
||||
-{
|
||||
- struct device_node *layout_np;
|
||||
- struct nvmem_layout *l, *layout = ERR_PTR(-EPROBE_DEFER);
|
||||
-
|
||||
- layout_np = of_nvmem_layout_get_container(nvmem);
|
||||
- if (!layout_np)
|
||||
- return NULL;
|
||||
-
|
||||
- /* Fixed layouts don't have a matching driver */
|
||||
- if (of_device_is_compatible(layout_np, "fixed-layout")) {
|
||||
- of_node_put(layout_np);
|
||||
- return NULL;
|
||||
- }
|
||||
-
|
||||
- /*
|
||||
- * In case the nvmem device was built-in while the layout was built as a
|
||||
- * module, we shall manually request the layout driver loading otherwise
|
||||
- * we'll never have any match.
|
||||
- */
|
||||
- of_request_module(layout_np);
|
||||
-
|
||||
- spin_lock(&nvmem_layout_lock);
|
||||
-
|
||||
- list_for_each_entry(l, &nvmem_layouts, node) {
|
||||
- if (of_match_node(l->of_match_table, layout_np)) {
|
||||
- if (try_module_get(l->owner))
|
||||
- layout = l;
|
||||
-
|
||||
- break;
|
||||
- }
|
||||
- }
|
||||
-
|
||||
- spin_unlock(&nvmem_layout_lock);
|
||||
- of_node_put(layout_np);
|
||||
-
|
||||
- return layout;
|
||||
-}
|
||||
-
|
||||
-static void nvmem_layout_put(struct nvmem_layout *layout)
|
||||
-{
|
||||
- if (layout)
|
||||
- module_put(layout->owner);
|
||||
-}
|
||||
-
|
||||
-static int nvmem_add_cells_from_layout(struct nvmem_device *nvmem)
|
||||
-{
|
||||
- struct nvmem_layout *layout = nvmem->layout;
|
||||
- int ret;
|
||||
-
|
||||
- if (layout && layout->add_cells) {
|
||||
- ret = layout->add_cells(&nvmem->dev, nvmem);
|
||||
- if (ret)
|
||||
- return ret;
|
||||
- }
|
||||
-
|
||||
- return 0;
|
||||
-}
|
||||
-
|
||||
-#if IS_ENABLED(CONFIG_OF)
|
||||
-struct device_node *of_nvmem_layout_get_container(struct nvmem_device *nvmem)
|
||||
-{
|
||||
- return of_get_child_by_name(nvmem->dev.of_node, "nvmem-layout");
|
||||
-}
|
||||
-EXPORT_SYMBOL_GPL(of_nvmem_layout_get_container);
|
||||
-#endif
|
||||
-
|
||||
const void *nvmem_layout_get_match_data(struct nvmem_device *nvmem,
|
||||
struct nvmem_layout *layout)
|
||||
{
|
||||
@@ -838,7 +760,7 @@ const void *nvmem_layout_get_match_data(
|
||||
const struct of_device_id *match;
|
||||
|
||||
layout_np = of_nvmem_layout_get_container(nvmem);
|
||||
- match = of_match_node(layout->of_match_table, layout_np);
|
||||
+ match = of_match_node(layout->dev.driver->of_match_table, layout_np);
|
||||
|
||||
return match ? match->data : NULL;
|
||||
}
|
||||
@@ -950,19 +872,6 @@ struct nvmem_device *nvmem_register(cons
|
||||
goto err_put_device;
|
||||
}
|
||||
|
||||
- /*
|
||||
- * If the driver supplied a layout by config->layout, the module
|
||||
- * pointer will be NULL and nvmem_layout_put() will be a noop.
|
||||
- */
|
||||
- nvmem->layout = config->layout ?: nvmem_layout_get(nvmem);
|
||||
- if (IS_ERR(nvmem->layout)) {
|
||||
- rval = PTR_ERR(nvmem->layout);
|
||||
- nvmem->layout = NULL;
|
||||
-
|
||||
- if (rval == -EPROBE_DEFER)
|
||||
- goto err_teardown_compat;
|
||||
- }
|
||||
-
|
||||
if (config->cells) {
|
||||
rval = nvmem_add_cells(nvmem, config->cells, config->ncells);
|
||||
if (rval)
|
||||
@@ -983,24 +892,24 @@ struct nvmem_device *nvmem_register(cons
|
||||
if (rval)
|
||||
goto err_remove_cells;
|
||||
|
||||
- rval = nvmem_add_cells_from_layout(nvmem);
|
||||
- if (rval)
|
||||
- goto err_remove_cells;
|
||||
-
|
||||
dev_dbg(&nvmem->dev, "Registering nvmem device %s\n", config->name);
|
||||
|
||||
rval = device_add(&nvmem->dev);
|
||||
if (rval)
|
||||
goto err_remove_cells;
|
||||
|
||||
+ rval = nvmem_populate_layout(nvmem);
|
||||
+ if (rval)
|
||||
+ goto err_remove_dev;
|
||||
+
|
||||
blocking_notifier_call_chain(&nvmem_notifier, NVMEM_ADD, nvmem);
|
||||
|
||||
return nvmem;
|
||||
|
||||
+err_remove_dev:
|
||||
+ device_del(&nvmem->dev);
|
||||
err_remove_cells:
|
||||
nvmem_device_remove_all_cells(nvmem);
|
||||
- nvmem_layout_put(nvmem->layout);
|
||||
-err_teardown_compat:
|
||||
if (config->compat)
|
||||
nvmem_sysfs_remove_compat(nvmem, config);
|
||||
err_put_device:
|
||||
@@ -1022,7 +931,7 @@ static void nvmem_device_release(struct
|
||||
device_remove_bin_file(nvmem->base_dev, &nvmem->eeprom);
|
||||
|
||||
nvmem_device_remove_all_cells(nvmem);
|
||||
- nvmem_layout_put(nvmem->layout);
|
||||
+ nvmem_destroy_layout(nvmem);
|
||||
device_unregister(&nvmem->dev);
|
||||
}
|
||||
|
||||
@@ -1324,6 +1233,12 @@ nvmem_cell_get_from_lookup(struct device
|
||||
return cell;
|
||||
}
|
||||
|
||||
+static void nvmem_layout_module_put(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ if (nvmem->layout && nvmem->layout->dev.driver)
|
||||
+ module_put(nvmem->layout->dev.driver->owner);
|
||||
+}
|
||||
+
|
||||
#if IS_ENABLED(CONFIG_OF)
|
||||
static struct nvmem_cell_entry *
|
||||
nvmem_find_cell_entry_by_node(struct nvmem_device *nvmem, struct device_node *np)
|
||||
@@ -1342,6 +1257,18 @@ nvmem_find_cell_entry_by_node(struct nvm
|
||||
return cell;
|
||||
}
|
||||
|
||||
+static int nvmem_layout_module_get_optional(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ if (!nvmem->layout)
|
||||
+ return 0;
|
||||
+
|
||||
+ if (!nvmem->layout->dev.driver ||
|
||||
+ !try_module_get(nvmem->layout->dev.driver->owner))
|
||||
+ return -EPROBE_DEFER;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
/**
|
||||
* of_nvmem_cell_get() - Get a nvmem cell from given device node and cell id
|
||||
*
|
||||
@@ -1404,16 +1331,29 @@ struct nvmem_cell *of_nvmem_cell_get(str
|
||||
return ERR_CAST(nvmem);
|
||||
}
|
||||
|
||||
+ ret = nvmem_layout_module_get_optional(nvmem);
|
||||
+ if (ret) {
|
||||
+ of_node_put(cell_np);
|
||||
+ __nvmem_device_put(nvmem);
|
||||
+ return ERR_PTR(ret);
|
||||
+ }
|
||||
+
|
||||
cell_entry = nvmem_find_cell_entry_by_node(nvmem, cell_np);
|
||||
of_node_put(cell_np);
|
||||
if (!cell_entry) {
|
||||
__nvmem_device_put(nvmem);
|
||||
- return ERR_PTR(-ENOENT);
|
||||
+ nvmem_layout_module_put(nvmem);
|
||||
+ if (nvmem->layout)
|
||||
+ return ERR_PTR(-EPROBE_DEFER);
|
||||
+ else
|
||||
+ return ERR_PTR(-ENOENT);
|
||||
}
|
||||
|
||||
cell = nvmem_create_cell(cell_entry, id, cell_index);
|
||||
- if (IS_ERR(cell))
|
||||
+ if (IS_ERR(cell)) {
|
||||
__nvmem_device_put(nvmem);
|
||||
+ nvmem_layout_module_put(nvmem);
|
||||
+ }
|
||||
|
||||
return cell;
|
||||
}
|
||||
@@ -1527,6 +1467,7 @@ void nvmem_cell_put(struct nvmem_cell *c
|
||||
|
||||
kfree(cell);
|
||||
__nvmem_device_put(nvmem);
|
||||
+ nvmem_layout_module_put(nvmem);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(nvmem_cell_put);
|
||||
|
||||
@@ -2104,11 +2045,22 @@ EXPORT_SYMBOL_GPL(nvmem_dev_name);
|
||||
|
||||
static int __init nvmem_init(void)
|
||||
{
|
||||
- return bus_register(&nvmem_bus_type);
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = bus_register(&nvmem_bus_type);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ ret = nvmem_layout_bus_register();
|
||||
+ if (ret)
|
||||
+ bus_unregister(&nvmem_bus_type);
|
||||
+
|
||||
+ return ret;
|
||||
}
|
||||
|
||||
static void __exit nvmem_exit(void)
|
||||
{
|
||||
+ nvmem_layout_bus_unregister();
|
||||
bus_unregister(&nvmem_bus_type);
|
||||
}
|
||||
|
||||
--- a/drivers/nvmem/internals.h
|
||||
+++ b/drivers/nvmem/internals.h
|
||||
@@ -34,4 +34,25 @@ struct nvmem_device {
|
||||
void *priv;
|
||||
};
|
||||
|
||||
+#if IS_ENABLED(CONFIG_OF)
|
||||
+int nvmem_layout_bus_register(void);
|
||||
+void nvmem_layout_bus_unregister(void);
|
||||
+int nvmem_populate_layout(struct nvmem_device *nvmem);
|
||||
+void nvmem_destroy_layout(struct nvmem_device *nvmem);
|
||||
+#else /* CONFIG_OF */
|
||||
+static inline int nvmem_layout_bus_register(void)
|
||||
+{
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static inline void nvmem_layout_bus_unregister(void) {}
|
||||
+
|
||||
+static inline int nvmem_populate_layout(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static inline void nvmem_destroy_layout(struct nvmem_device *nvmem) { }
|
||||
+#endif /* CONFIG_OF */
|
||||
+
|
||||
#endif /* ifndef _LINUX_NVMEM_INTERNALS_H */
|
||||
--- /dev/null
|
||||
+++ b/drivers/nvmem/layouts.c
|
||||
@@ -0,0 +1,201 @@
|
||||
+// SPDX-License-Identifier: GPL-2.0
|
||||
+/*
|
||||
+ * NVMEM layout bus handling
|
||||
+ *
|
||||
+ * Copyright (C) 2023 Bootlin
|
||||
+ * Author: Miquel Raynal <miquel.raynal@bootlin.com
|
||||
+ */
|
||||
+
|
||||
+#include <linux/device.h>
|
||||
+#include <linux/dma-mapping.h>
|
||||
+#include <linux/nvmem-consumer.h>
|
||||
+#include <linux/nvmem-provider.h>
|
||||
+#include <linux/of.h>
|
||||
+#include <linux/of_device.h>
|
||||
+#include <linux/of_irq.h>
|
||||
+
|
||||
+#include "internals.h"
|
||||
+
|
||||
+#define to_nvmem_layout_driver(drv) \
|
||||
+ (container_of((drv), struct nvmem_layout_driver, driver))
|
||||
+#define to_nvmem_layout_device(_dev) \
|
||||
+ container_of((_dev), struct nvmem_layout, dev)
|
||||
+
|
||||
+static int nvmem_layout_bus_match(struct device *dev, struct device_driver *drv)
|
||||
+{
|
||||
+ return of_driver_match_device(dev, drv);
|
||||
+}
|
||||
+
|
||||
+static int nvmem_layout_bus_probe(struct device *dev)
|
||||
+{
|
||||
+ struct nvmem_layout_driver *drv = to_nvmem_layout_driver(dev->driver);
|
||||
+ struct nvmem_layout *layout = to_nvmem_layout_device(dev);
|
||||
+
|
||||
+ if (!drv->probe || !drv->remove)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ return drv->probe(layout);
|
||||
+}
|
||||
+
|
||||
+static void nvmem_layout_bus_remove(struct device *dev)
|
||||
+{
|
||||
+ struct nvmem_layout_driver *drv = to_nvmem_layout_driver(dev->driver);
|
||||
+ struct nvmem_layout *layout = to_nvmem_layout_device(dev);
|
||||
+
|
||||
+ return drv->remove(layout);
|
||||
+}
|
||||
+
|
||||
+static struct bus_type nvmem_layout_bus_type = {
|
||||
+ .name = "nvmem-layout",
|
||||
+ .match = nvmem_layout_bus_match,
|
||||
+ .probe = nvmem_layout_bus_probe,
|
||||
+ .remove = nvmem_layout_bus_remove,
|
||||
+};
|
||||
+
|
||||
+int nvmem_layout_driver_register(struct nvmem_layout_driver *drv)
|
||||
+{
|
||||
+ drv->driver.bus = &nvmem_layout_bus_type;
|
||||
+
|
||||
+ return driver_register(&drv->driver);
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(nvmem_layout_driver_register);
|
||||
+
|
||||
+void nvmem_layout_driver_unregister(struct nvmem_layout_driver *drv)
|
||||
+{
|
||||
+ driver_unregister(&drv->driver);
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(nvmem_layout_driver_unregister);
|
||||
+
|
||||
+static void nvmem_layout_release_device(struct device *dev)
|
||||
+{
|
||||
+ struct nvmem_layout *layout = to_nvmem_layout_device(dev);
|
||||
+
|
||||
+ of_node_put(layout->dev.of_node);
|
||||
+ kfree(layout);
|
||||
+}
|
||||
+
|
||||
+static int nvmem_layout_create_device(struct nvmem_device *nvmem,
|
||||
+ struct device_node *np)
|
||||
+{
|
||||
+ struct nvmem_layout *layout;
|
||||
+ struct device *dev;
|
||||
+ int ret;
|
||||
+
|
||||
+ layout = kzalloc(sizeof(*layout), GFP_KERNEL);
|
||||
+ if (!layout)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ /* Create a bidirectional link */
|
||||
+ layout->nvmem = nvmem;
|
||||
+ nvmem->layout = layout;
|
||||
+
|
||||
+ /* Device model registration */
|
||||
+ dev = &layout->dev;
|
||||
+ device_initialize(dev);
|
||||
+ dev->parent = &nvmem->dev;
|
||||
+ dev->bus = &nvmem_layout_bus_type;
|
||||
+ dev->release = nvmem_layout_release_device;
|
||||
+ dev->coherent_dma_mask = DMA_BIT_MASK(32);
|
||||
+ dev->dma_mask = &dev->coherent_dma_mask;
|
||||
+ device_set_node(dev, of_fwnode_handle(of_node_get(np)));
|
||||
+ of_device_make_bus_id(dev);
|
||||
+ of_msi_configure(dev, dev->of_node);
|
||||
+
|
||||
+ ret = device_add(dev);
|
||||
+ if (ret) {
|
||||
+ put_device(dev);
|
||||
+ return ret;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static const struct of_device_id of_nvmem_layout_skip_table[] = {
|
||||
+ { .compatible = "fixed-layout", },
|
||||
+ {}
|
||||
+};
|
||||
+
|
||||
+static int nvmem_layout_bus_populate(struct nvmem_device *nvmem,
|
||||
+ struct device_node *layout_dn)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ /* Make sure it has a compatible property */
|
||||
+ if (!of_get_property(layout_dn, "compatible", NULL)) {
|
||||
+ pr_debug("%s() - skipping %pOF, no compatible prop\n",
|
||||
+ __func__, layout_dn);
|
||||
+ return 0;
|
||||
+ }
|
||||
+
|
||||
+ /* Fixed layouts are parsed manually somewhere else for now */
|
||||
+ if (of_match_node(of_nvmem_layout_skip_table, layout_dn)) {
|
||||
+ pr_debug("%s() - skipping %pOF node\n", __func__, layout_dn);
|
||||
+ return 0;
|
||||
+ }
|
||||
+
|
||||
+ if (of_node_check_flag(layout_dn, OF_POPULATED_BUS)) {
|
||||
+ pr_debug("%s() - skipping %pOF, already populated\n",
|
||||
+ __func__, layout_dn);
|
||||
+
|
||||
+ return 0;
|
||||
+ }
|
||||
+
|
||||
+ /* NVMEM layout buses expect only a single device representing the layout */
|
||||
+ ret = nvmem_layout_create_device(nvmem, layout_dn);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ of_node_set_flag(layout_dn, OF_POPULATED_BUS);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+struct device_node *of_nvmem_layout_get_container(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ return of_get_child_by_name(nvmem->dev.of_node, "nvmem-layout");
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(of_nvmem_layout_get_container);
|
||||
+
|
||||
+/*
|
||||
+ * Returns the number of devices populated, 0 if the operation was not relevant
|
||||
+ * for this nvmem device, an error code otherwise.
|
||||
+ */
|
||||
+int nvmem_populate_layout(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ struct device_node *layout_dn;
|
||||
+ int ret;
|
||||
+
|
||||
+ layout_dn = of_nvmem_layout_get_container(nvmem);
|
||||
+ if (!layout_dn)
|
||||
+ return 0;
|
||||
+
|
||||
+ /* Populate the layout device */
|
||||
+ device_links_supplier_sync_state_pause();
|
||||
+ ret = nvmem_layout_bus_populate(nvmem, layout_dn);
|
||||
+ device_links_supplier_sync_state_resume();
|
||||
+
|
||||
+ of_node_put(layout_dn);
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
+void nvmem_destroy_layout(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ struct device *dev;
|
||||
+
|
||||
+ if (!nvmem->layout)
|
||||
+ return;
|
||||
+
|
||||
+ dev = &nvmem->layout->dev;
|
||||
+ of_node_clear_flag(dev->of_node, OF_POPULATED_BUS);
|
||||
+ device_unregister(dev);
|
||||
+}
|
||||
+
|
||||
+int nvmem_layout_bus_register(void)
|
||||
+{
|
||||
+ return bus_register(&nvmem_layout_bus_type);
|
||||
+}
|
||||
+
|
||||
+void nvmem_layout_bus_unregister(void)
|
||||
+{
|
||||
+ bus_unregister(&nvmem_layout_bus_type);
|
||||
+}
|
||||
--- a/drivers/nvmem/layouts/Kconfig
|
||||
+++ b/drivers/nvmem/layouts/Kconfig
|
||||
@@ -1,5 +1,11 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
|
||||
+config NVMEM_LAYOUTS
|
||||
+ bool
|
||||
+ depends on OF
|
||||
+
|
||||
+if NVMEM_LAYOUTS
|
||||
+
|
||||
menu "Layout Types"
|
||||
|
||||
config NVMEM_LAYOUT_SL28_VPD
|
||||
@@ -21,3 +27,5 @@ config NVMEM_LAYOUT_ONIE_TLV
|
||||
If unsure, say N.
|
||||
|
||||
endmenu
|
||||
+
|
||||
+endif
|
||||
--- a/drivers/nvmem/layouts/onie-tlv.c
|
||||
+++ b/drivers/nvmem/layouts/onie-tlv.c
|
||||
@@ -225,16 +225,32 @@ static int onie_tlv_parse_table(struct d
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int onie_tlv_probe(struct nvmem_layout *layout)
|
||||
+{
|
||||
+ layout->add_cells = onie_tlv_parse_table;
|
||||
+
|
||||
+ return nvmem_layout_register(layout);
|
||||
+}
|
||||
+
|
||||
+static void onie_tlv_remove(struct nvmem_layout *layout)
|
||||
+{
|
||||
+ nvmem_layout_unregister(layout);
|
||||
+}
|
||||
+
|
||||
static const struct of_device_id onie_tlv_of_match_table[] = {
|
||||
{ .compatible = "onie,tlv-layout", },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, onie_tlv_of_match_table);
|
||||
|
||||
-static struct nvmem_layout onie_tlv_layout = {
|
||||
- .name = "ONIE tlv layout",
|
||||
- .of_match_table = onie_tlv_of_match_table,
|
||||
- .add_cells = onie_tlv_parse_table,
|
||||
+static struct nvmem_layout_driver onie_tlv_layout = {
|
||||
+ .driver = {
|
||||
+ .owner = THIS_MODULE,
|
||||
+ .name = "onie-tlv-layout",
|
||||
+ .of_match_table = onie_tlv_of_match_table,
|
||||
+ },
|
||||
+ .probe = onie_tlv_probe,
|
||||
+ .remove = onie_tlv_remove,
|
||||
};
|
||||
module_nvmem_layout_driver(onie_tlv_layout);
|
||||
|
||||
--- a/drivers/nvmem/layouts/sl28vpd.c
|
||||
+++ b/drivers/nvmem/layouts/sl28vpd.c
|
||||
@@ -134,16 +134,32 @@ static int sl28vpd_add_cells(struct devi
|
||||
return 0;
|
||||
}
|
||||
|
||||
+static int sl28vpd_probe(struct nvmem_layout *layout)
|
||||
+{
|
||||
+ layout->add_cells = sl28vpd_add_cells;
|
||||
+
|
||||
+ return nvmem_layout_register(layout);
|
||||
+}
|
||||
+
|
||||
+static void sl28vpd_remove(struct nvmem_layout *layout)
|
||||
+{
|
||||
+ nvmem_layout_unregister(layout);
|
||||
+}
|
||||
+
|
||||
static const struct of_device_id sl28vpd_of_match_table[] = {
|
||||
{ .compatible = "kontron,sl28-vpd" },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, sl28vpd_of_match_table);
|
||||
|
||||
-static struct nvmem_layout sl28vpd_layout = {
|
||||
- .name = "sl28-vpd",
|
||||
- .of_match_table = sl28vpd_of_match_table,
|
||||
- .add_cells = sl28vpd_add_cells,
|
||||
+static struct nvmem_layout_driver sl28vpd_layout = {
|
||||
+ .driver = {
|
||||
+ .owner = THIS_MODULE,
|
||||
+ .name = "kontron-sl28vpd-layout",
|
||||
+ .of_match_table = sl28vpd_of_match_table,
|
||||
+ },
|
||||
+ .probe = sl28vpd_probe,
|
||||
+ .remove = sl28vpd_remove,
|
||||
};
|
||||
module_nvmem_layout_driver(sl28vpd_layout);
|
||||
|
||||
--- a/include/linux/nvmem-provider.h
|
||||
+++ b/include/linux/nvmem-provider.h
|
||||
@@ -9,6 +9,7 @@
|
||||
#ifndef _LINUX_NVMEM_PROVIDER_H
|
||||
#define _LINUX_NVMEM_PROVIDER_H
|
||||
|
||||
+#include <linux/device.h>
|
||||
#include <linux/device/driver.h>
|
||||
#include <linux/err.h>
|
||||
#include <linux/errno.h>
|
||||
@@ -158,12 +159,11 @@ struct nvmem_cell_table {
|
||||
/**
|
||||
* struct nvmem_layout - NVMEM layout definitions
|
||||
*
|
||||
- * @name: Layout name.
|
||||
- * @of_match_table: Open firmware match table.
|
||||
- * @add_cells: Called to populate the layout using
|
||||
- * nvmem_add_one_cell().
|
||||
- * @owner: Pointer to struct module.
|
||||
- * @node: List node.
|
||||
+ * @dev: Device-model layout device.
|
||||
+ * @nvmem: The underlying NVMEM device
|
||||
+ * @add_cells: Will be called if a nvmem device is found which
|
||||
+ * has this layout. The function will add layout
|
||||
+ * specific cells with nvmem_add_one_cell().
|
||||
*
|
||||
* A nvmem device can hold a well defined structure which can just be
|
||||
* evaluated during runtime. For example a TLV list, or a list of "name=val"
|
||||
@@ -171,13 +171,15 @@ struct nvmem_cell_table {
|
||||
* cells.
|
||||
*/
|
||||
struct nvmem_layout {
|
||||
- const char *name;
|
||||
- const struct of_device_id *of_match_table;
|
||||
+ struct device dev;
|
||||
+ struct nvmem_device *nvmem;
|
||||
int (*add_cells)(struct device *dev, struct nvmem_device *nvmem);
|
||||
+};
|
||||
|
||||
- /* private */
|
||||
- struct module *owner;
|
||||
- struct list_head node;
|
||||
+struct nvmem_layout_driver {
|
||||
+ struct device_driver driver;
|
||||
+ int (*probe)(struct nvmem_layout *layout);
|
||||
+ void (*remove)(struct nvmem_layout *layout);
|
||||
};
|
||||
|
||||
#if IS_ENABLED(CONFIG_NVMEM)
|
||||
@@ -194,11 +196,15 @@ void nvmem_del_cell_table(struct nvmem_c
|
||||
int nvmem_add_one_cell(struct nvmem_device *nvmem,
|
||||
const struct nvmem_cell_info *info);
|
||||
|
||||
-int __nvmem_layout_register(struct nvmem_layout *layout, struct module *owner);
|
||||
-#define nvmem_layout_register(layout) \
|
||||
- __nvmem_layout_register(layout, THIS_MODULE)
|
||||
+int nvmem_layout_register(struct nvmem_layout *layout);
|
||||
void nvmem_layout_unregister(struct nvmem_layout *layout);
|
||||
|
||||
+int nvmem_layout_driver_register(struct nvmem_layout_driver *drv);
|
||||
+void nvmem_layout_driver_unregister(struct nvmem_layout_driver *drv);
|
||||
+#define module_nvmem_layout_driver(__nvmem_layout_driver) \
|
||||
+ module_driver(__nvmem_layout_driver, nvmem_layout_driver_register, \
|
||||
+ nvmem_layout_driver_unregister)
|
||||
+
|
||||
const void *nvmem_layout_get_match_data(struct nvmem_device *nvmem,
|
||||
struct nvmem_layout *layout);
|
||||
|
||||
@@ -262,8 +268,4 @@ static inline struct device_node *of_nvm
|
||||
|
||||
#endif /* CONFIG_NVMEM && CONFIG_OF */
|
||||
|
||||
-#define module_nvmem_layout_driver(__layout_driver) \
|
||||
- module_driver(__layout_driver, nvmem_layout_register, \
|
||||
- nvmem_layout_unregister)
|
||||
-
|
||||
#endif /* ifndef _LINUX_NVMEM_PROVIDER_H */
|
@ -0,0 +1,240 @@
|
||||
From 0331c611949fffdf486652450901a4dc52bc5cca Mon Sep 17 00:00:00 2001
|
||||
From: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Date: Fri, 15 Dec 2023 11:15:34 +0000
|
||||
Subject: [PATCH] nvmem: core: Expose cells through sysfs
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
The binary content of nvmem devices is available to the user so in the
|
||||
easiest cases, finding the content of a cell is rather easy as it is
|
||||
just a matter of looking at a known and fixed offset. However, nvmem
|
||||
layouts have been recently introduced to cope with more advanced
|
||||
situations, where the offset and size of the cells is not known in
|
||||
advance or is dynamic. When using layouts, more advanced parsers are
|
||||
used by the kernel in order to give direct access to the content of each
|
||||
cell, regardless of its position/size in the underlying
|
||||
device. Unfortunately, these information are not accessible by users,
|
||||
unless by fully re-implementing the parser logic in userland.
|
||||
|
||||
Let's expose the cells and their content through sysfs to avoid these
|
||||
situations. Of course the relevant NVMEM sysfs Kconfig option must be
|
||||
enabled for this support to be available.
|
||||
|
||||
Not all nvmem devices expose cells. Indeed, the .bin_attrs attribute
|
||||
group member will be filled at runtime only when relevant and will
|
||||
remain empty otherwise. In this case, as the cells attribute group will
|
||||
be empty, it will not lead to any additional folder/file creation.
|
||||
|
||||
Exposed cells are read-only. There is, in practice, everything in the
|
||||
core to support a write path, but as I don't see any need for that, I
|
||||
prefer to keep the interface simple (and probably safer). The interface
|
||||
is documented as being in the "testing" state which means we can later
|
||||
add a write attribute if though relevant.
|
||||
|
||||
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Tested-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Tested-by: Chen-Yu Tsai <wenst@chromium.org>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231215111536.316972-9-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/core.c | 135 +++++++++++++++++++++++++++++++++++++-
|
||||
drivers/nvmem/internals.h | 1 +
|
||||
2 files changed, 135 insertions(+), 1 deletion(-)
|
||||
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -299,6 +299,43 @@ static umode_t nvmem_bin_attr_is_visible
|
||||
return nvmem_bin_attr_get_umode(nvmem);
|
||||
}
|
||||
|
||||
+static struct nvmem_cell *nvmem_create_cell(struct nvmem_cell_entry *entry,
|
||||
+ const char *id, int index);
|
||||
+
|
||||
+static ssize_t nvmem_cell_attr_read(struct file *filp, struct kobject *kobj,
|
||||
+ struct bin_attribute *attr, char *buf,
|
||||
+ loff_t pos, size_t count)
|
||||
+{
|
||||
+ struct nvmem_cell_entry *entry;
|
||||
+ struct nvmem_cell *cell = NULL;
|
||||
+ size_t cell_sz, read_len;
|
||||
+ void *content;
|
||||
+
|
||||
+ entry = attr->private;
|
||||
+ cell = nvmem_create_cell(entry, entry->name, 0);
|
||||
+ if (IS_ERR(cell))
|
||||
+ return PTR_ERR(cell);
|
||||
+
|
||||
+ if (!cell)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ content = nvmem_cell_read(cell, &cell_sz);
|
||||
+ if (IS_ERR(content)) {
|
||||
+ read_len = PTR_ERR(content);
|
||||
+ goto destroy_cell;
|
||||
+ }
|
||||
+
|
||||
+ read_len = min_t(unsigned int, cell_sz - pos, count);
|
||||
+ memcpy(buf, content + pos, read_len);
|
||||
+ kfree(content);
|
||||
+
|
||||
+destroy_cell:
|
||||
+ kfree_const(cell->id);
|
||||
+ kfree(cell);
|
||||
+
|
||||
+ return read_len;
|
||||
+}
|
||||
+
|
||||
/* default read/write permissions */
|
||||
static struct bin_attribute bin_attr_rw_nvmem = {
|
||||
.attr = {
|
||||
@@ -320,11 +357,21 @@ static const struct attribute_group nvme
|
||||
.is_bin_visible = nvmem_bin_attr_is_visible,
|
||||
};
|
||||
|
||||
+/* Cell attributes will be dynamically allocated */
|
||||
+static struct attribute_group nvmem_cells_group = {
|
||||
+ .name = "cells",
|
||||
+};
|
||||
+
|
||||
static const struct attribute_group *nvmem_dev_groups[] = {
|
||||
&nvmem_bin_group,
|
||||
NULL,
|
||||
};
|
||||
|
||||
+static const struct attribute_group *nvmem_cells_groups[] = {
|
||||
+ &nvmem_cells_group,
|
||||
+ NULL,
|
||||
+};
|
||||
+
|
||||
static struct bin_attribute bin_attr_nvmem_eeprom_compat = {
|
||||
.attr = {
|
||||
.name = "eeprom",
|
||||
@@ -380,6 +427,68 @@ static void nvmem_sysfs_remove_compat(st
|
||||
device_remove_bin_file(nvmem->base_dev, &nvmem->eeprom);
|
||||
}
|
||||
|
||||
+static int nvmem_populate_sysfs_cells(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ struct bin_attribute **cells_attrs, *attrs;
|
||||
+ struct nvmem_cell_entry *entry;
|
||||
+ unsigned int ncells = 0, i = 0;
|
||||
+ int ret = 0;
|
||||
+
|
||||
+ mutex_lock(&nvmem_mutex);
|
||||
+
|
||||
+ if (list_empty(&nvmem->cells) || nvmem->sysfs_cells_populated) {
|
||||
+ nvmem_cells_group.bin_attrs = NULL;
|
||||
+ goto unlock_mutex;
|
||||
+ }
|
||||
+
|
||||
+ /* Allocate an array of attributes with a sentinel */
|
||||
+ ncells = list_count_nodes(&nvmem->cells);
|
||||
+ cells_attrs = devm_kcalloc(&nvmem->dev, ncells + 1,
|
||||
+ sizeof(struct bin_attribute *), GFP_KERNEL);
|
||||
+ if (!cells_attrs) {
|
||||
+ ret = -ENOMEM;
|
||||
+ goto unlock_mutex;
|
||||
+ }
|
||||
+
|
||||
+ attrs = devm_kcalloc(&nvmem->dev, ncells, sizeof(struct bin_attribute), GFP_KERNEL);
|
||||
+ if (!attrs) {
|
||||
+ ret = -ENOMEM;
|
||||
+ goto unlock_mutex;
|
||||
+ }
|
||||
+
|
||||
+ /* Initialize each attribute to take the name and size of the cell */
|
||||
+ list_for_each_entry(entry, &nvmem->cells, node) {
|
||||
+ sysfs_bin_attr_init(&attrs[i]);
|
||||
+ attrs[i].attr.name = devm_kasprintf(&nvmem->dev, GFP_KERNEL,
|
||||
+ "%s@%x", entry->name,
|
||||
+ entry->offset);
|
||||
+ attrs[i].attr.mode = 0444;
|
||||
+ attrs[i].size = entry->bytes;
|
||||
+ attrs[i].read = &nvmem_cell_attr_read;
|
||||
+ attrs[i].private = entry;
|
||||
+ if (!attrs[i].attr.name) {
|
||||
+ ret = -ENOMEM;
|
||||
+ goto unlock_mutex;
|
||||
+ }
|
||||
+
|
||||
+ cells_attrs[i] = &attrs[i];
|
||||
+ i++;
|
||||
+ }
|
||||
+
|
||||
+ nvmem_cells_group.bin_attrs = cells_attrs;
|
||||
+
|
||||
+ ret = devm_device_add_groups(&nvmem->dev, nvmem_cells_groups);
|
||||
+ if (ret)
|
||||
+ goto unlock_mutex;
|
||||
+
|
||||
+ nvmem->sysfs_cells_populated = true;
|
||||
+
|
||||
+unlock_mutex:
|
||||
+ mutex_unlock(&nvmem_mutex);
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
#else /* CONFIG_NVMEM_SYSFS */
|
||||
|
||||
static int nvmem_sysfs_setup_compat(struct nvmem_device *nvmem,
|
||||
@@ -739,11 +848,25 @@ static int nvmem_add_cells_from_fixed_la
|
||||
|
||||
int nvmem_layout_register(struct nvmem_layout *layout)
|
||||
{
|
||||
+ int ret;
|
||||
+
|
||||
if (!layout->add_cells)
|
||||
return -EINVAL;
|
||||
|
||||
/* Populate the cells */
|
||||
- return layout->add_cells(&layout->nvmem->dev, layout->nvmem);
|
||||
+ ret = layout->add_cells(&layout->nvmem->dev, layout->nvmem);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+#ifdef CONFIG_NVMEM_SYSFS
|
||||
+ ret = nvmem_populate_sysfs_cells(layout->nvmem);
|
||||
+ if (ret) {
|
||||
+ nvmem_device_remove_all_cells(layout->nvmem);
|
||||
+ return ret;
|
||||
+ }
|
||||
+#endif
|
||||
+
|
||||
+ return 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(nvmem_layout_register);
|
||||
|
||||
@@ -902,10 +1025,20 @@ struct nvmem_device *nvmem_register(cons
|
||||
if (rval)
|
||||
goto err_remove_dev;
|
||||
|
||||
+#ifdef CONFIG_NVMEM_SYSFS
|
||||
+ rval = nvmem_populate_sysfs_cells(nvmem);
|
||||
+ if (rval)
|
||||
+ goto err_destroy_layout;
|
||||
+#endif
|
||||
+
|
||||
blocking_notifier_call_chain(&nvmem_notifier, NVMEM_ADD, nvmem);
|
||||
|
||||
return nvmem;
|
||||
|
||||
+#ifdef CONFIG_NVMEM_SYSFS
|
||||
+err_destroy_layout:
|
||||
+ nvmem_destroy_layout(nvmem);
|
||||
+#endif
|
||||
err_remove_dev:
|
||||
device_del(&nvmem->dev);
|
||||
err_remove_cells:
|
||||
--- a/drivers/nvmem/internals.h
|
||||
+++ b/drivers/nvmem/internals.h
|
||||
@@ -32,6 +32,7 @@ struct nvmem_device {
|
||||
struct gpio_desc *wp_gpio;
|
||||
struct nvmem_layout *layout;
|
||||
void *priv;
|
||||
+ bool sysfs_cells_populated;
|
||||
};
|
||||
|
||||
#if IS_ENABLED(CONFIG_OF)
|
@ -0,0 +1,65 @@
|
||||
From f0ac5b23039610619ca4a4805528553ecb6bc815 Mon Sep 17 00:00:00 2001
|
||||
From: Patrick Delaunay <patrick.delaunay@foss.st.com>
|
||||
Date: Fri, 15 Dec 2023 11:15:36 +0000
|
||||
Subject: [PATCH] nvmem: stm32: add support for STM32MP25 BSEC to control OTP
|
||||
data
|
||||
|
||||
On STM32MP25, OTP area may be read/written by using BSEC (boot, security
|
||||
and OTP control). The BSEC internal peripheral is only managed by the
|
||||
secure world.
|
||||
|
||||
The 12 Kbits of OTP (effective) are organized into the following regions:
|
||||
- lower OTP (OTP0 to OTP127) = 4096 lower OTP bits,
|
||||
bitwise (1-bit) programmable
|
||||
- mid OTP (OTP128 to OTP255) = 4096 middle OTP bits,
|
||||
bulk (32-bit) programmable
|
||||
- upper OTP (OTP256 to OTP383) = 4096 upper OTP bits,
|
||||
bulk (32-bit) programmable,
|
||||
only accessible when BSEC is in closed state.
|
||||
|
||||
As HWKEY and ECIES key are only accessible by ROM code;
|
||||
only 368 OTP words are managed in this driver (OTP0 to OTP267).
|
||||
|
||||
This patch adds the STM32MP25 configuration for reading and writing
|
||||
the OTP data using the OP-TEE BSEC TA services.
|
||||
|
||||
Signed-off-by: Patrick Delaunay <patrick.delaunay@foss.st.com>
|
||||
Signed-off-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
|
||||
Link: https://lore.kernel.org/r/20231215111536.316972-11-srinivas.kandagatla@linaro.org
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/stm32-romem.c | 16 ++++++++++++++++
|
||||
1 file changed, 16 insertions(+)
|
||||
|
||||
--- a/drivers/nvmem/stm32-romem.c
|
||||
+++ b/drivers/nvmem/stm32-romem.c
|
||||
@@ -269,6 +269,19 @@ static const struct stm32_romem_cfg stm3
|
||||
.ta = true,
|
||||
};
|
||||
|
||||
+/*
|
||||
+ * STM32MP25 BSEC OTP: 3 regions of 32-bits data words
|
||||
+ * lower OTP (OTP0 to OTP127), bitwise (1-bit) programmable
|
||||
+ * mid OTP (OTP128 to OTP255), bulk (32-bit) programmable
|
||||
+ * upper OTP (OTP256 to OTP383), bulk (32-bit) programmable
|
||||
+ * but no access to HWKEY and ECIES key: limited at OTP367
|
||||
+ */
|
||||
+static const struct stm32_romem_cfg stm32mp25_bsec_cfg = {
|
||||
+ .size = 368 * 4,
|
||||
+ .lower = 127,
|
||||
+ .ta = true,
|
||||
+};
|
||||
+
|
||||
static const struct of_device_id stm32_romem_of_match[] __maybe_unused = {
|
||||
{ .compatible = "st,stm32f4-otp", }, {
|
||||
.compatible = "st,stm32mp15-bsec",
|
||||
@@ -276,6 +289,9 @@ static const struct of_device_id stm32_r
|
||||
}, {
|
||||
.compatible = "st,stm32mp13-bsec",
|
||||
.data = (void *)&stm32mp13_bsec_cfg,
|
||||
+ }, {
|
||||
+ .compatible = "st,stm32mp25-bsec",
|
||||
+ .data = (void *)&stm32mp25_bsec_cfg,
|
||||
},
|
||||
{ /* sentinel */ },
|
||||
};
|
@ -0,0 +1,94 @@
|
||||
From 401df0d4f4098ecc9c5278da2f50756d62e5b37d Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
|
||||
Date: Tue, 19 Dec 2023 13:01:03 +0100
|
||||
Subject: [PATCH] nvmem: layouts: refactor .add_cells() callback arguments
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
Simply pass whole "struct nvmem_layout" instead of single variables.
|
||||
There is nothing in "struct nvmem_layout" that we have to hide from
|
||||
layout drivers. They also access it during .probe() and .remove().
|
||||
|
||||
Thanks to this change:
|
||||
|
||||
1. API gets more consistent
|
||||
All layouts drivers callbacks get the same argument
|
||||
|
||||
2. Layouts get correct device
|
||||
Before this change NVMEM core code was passing NVMEM device instead
|
||||
of layout device. That resulted in:
|
||||
* Confusing prints
|
||||
* Calling devm_*() helpers on wrong device
|
||||
* Helpers like of_device_get_match_data() dereferencing NULLs
|
||||
|
||||
3. It gets possible to get match data
|
||||
First of all nvmem_layout_get_match_data() requires passing "struct
|
||||
nvmem_layout" which .add_cells() callback didn't have before this. It
|
||||
doesn't matter much as it's rather useless now anyway (and will be
|
||||
dropped).
|
||||
What's more important however is that of_device_get_match_data() can
|
||||
be used now thanks to owning a proper device pointer.
|
||||
|
||||
Signed-off-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Reviewed-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Reviewed-by: Michael Walle <michael@walle.cc>
|
||||
Link: https://lore.kernel.org/r/20231219120104.3422-1-zajec5@gmail.com
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/core.c | 2 +-
|
||||
drivers/nvmem/layouts/onie-tlv.c | 4 +++-
|
||||
drivers/nvmem/layouts/sl28vpd.c | 4 +++-
|
||||
include/linux/nvmem-provider.h | 2 +-
|
||||
4 files changed, 8 insertions(+), 4 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -854,7 +854,7 @@ int nvmem_layout_register(struct nvmem_l
|
||||
return -EINVAL;
|
||||
|
||||
/* Populate the cells */
|
||||
- ret = layout->add_cells(&layout->nvmem->dev, layout->nvmem);
|
||||
+ ret = layout->add_cells(layout);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
--- a/drivers/nvmem/layouts/onie-tlv.c
|
||||
+++ b/drivers/nvmem/layouts/onie-tlv.c
|
||||
@@ -182,8 +182,10 @@ static bool onie_tlv_crc_is_valid(struct
|
||||
return true;
|
||||
}
|
||||
|
||||
-static int onie_tlv_parse_table(struct device *dev, struct nvmem_device *nvmem)
|
||||
+static int onie_tlv_parse_table(struct nvmem_layout *layout)
|
||||
{
|
||||
+ struct nvmem_device *nvmem = layout->nvmem;
|
||||
+ struct device *dev = &layout->dev;
|
||||
struct onie_tlv_hdr hdr;
|
||||
size_t table_len, data_len, hdr_len;
|
||||
u8 *table, *data;
|
||||
--- a/drivers/nvmem/layouts/sl28vpd.c
|
||||
+++ b/drivers/nvmem/layouts/sl28vpd.c
|
||||
@@ -80,8 +80,10 @@ static int sl28vpd_v1_check_crc(struct d
|
||||
return 0;
|
||||
}
|
||||
|
||||
-static int sl28vpd_add_cells(struct device *dev, struct nvmem_device *nvmem)
|
||||
+static int sl28vpd_add_cells(struct nvmem_layout *layout)
|
||||
{
|
||||
+ struct nvmem_device *nvmem = layout->nvmem;
|
||||
+ struct device *dev = &layout->dev;
|
||||
const struct nvmem_cell_info *pinfo;
|
||||
struct nvmem_cell_info info = {0};
|
||||
struct device_node *layout_np;
|
||||
--- a/include/linux/nvmem-provider.h
|
||||
+++ b/include/linux/nvmem-provider.h
|
||||
@@ -173,7 +173,7 @@ struct nvmem_cell_table {
|
||||
struct nvmem_layout {
|
||||
struct device dev;
|
||||
struct nvmem_device *nvmem;
|
||||
- int (*add_cells)(struct device *dev, struct nvmem_device *nvmem);
|
||||
+ int (*add_cells)(struct nvmem_layout *layout);
|
||||
};
|
||||
|
||||
struct nvmem_layout_driver {
|
@ -0,0 +1,72 @@
|
||||
From 43f60e3fb62edc7bd8891de8779fb422f4ae23ae Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
|
||||
Date: Tue, 19 Dec 2023 13:01:04 +0100
|
||||
Subject: [PATCH] nvmem: drop nvmem_layout_get_match_data()
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
Thanks for layouts refactoring we now have "struct device" associated
|
||||
with layout. Also its OF pointer points directly to the "nvmem-layout"
|
||||
DT node.
|
||||
|
||||
All it takes to get match data is a generic of_device_get_match_data().
|
||||
|
||||
Signed-off-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Reviewed-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Reviewed-by: Michael Walle <michael@walle.cc>
|
||||
Link: https://lore.kernel.org/r/20231219120104.3422-2-zajec5@gmail.com
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/core.c | 13 -------------
|
||||
include/linux/nvmem-provider.h | 10 ----------
|
||||
2 files changed, 23 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -876,19 +876,6 @@ void nvmem_layout_unregister(struct nvme
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(nvmem_layout_unregister);
|
||||
|
||||
-const void *nvmem_layout_get_match_data(struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout)
|
||||
-{
|
||||
- struct device_node __maybe_unused *layout_np;
|
||||
- const struct of_device_id *match;
|
||||
-
|
||||
- layout_np = of_nvmem_layout_get_container(nvmem);
|
||||
- match = of_match_node(layout->dev.driver->of_match_table, layout_np);
|
||||
-
|
||||
- return match ? match->data : NULL;
|
||||
-}
|
||||
-EXPORT_SYMBOL_GPL(nvmem_layout_get_match_data);
|
||||
-
|
||||
/**
|
||||
* nvmem_register() - Register a nvmem device for given nvmem_config.
|
||||
* Also creates a binary entry in /sys/bus/nvmem/devices/dev-name/nvmem
|
||||
--- a/include/linux/nvmem-provider.h
|
||||
+++ b/include/linux/nvmem-provider.h
|
||||
@@ -205,9 +205,6 @@ void nvmem_layout_driver_unregister(stru
|
||||
module_driver(__nvmem_layout_driver, nvmem_layout_driver_register, \
|
||||
nvmem_layout_driver_unregister)
|
||||
|
||||
-const void *nvmem_layout_get_match_data(struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout);
|
||||
-
|
||||
#else
|
||||
|
||||
static inline struct nvmem_device *nvmem_register(const struct nvmem_config *c)
|
||||
@@ -238,13 +235,6 @@ static inline int nvmem_layout_register(
|
||||
|
||||
static inline void nvmem_layout_unregister(struct nvmem_layout *layout) {}
|
||||
|
||||
-static inline const void *
|
||||
-nvmem_layout_get_match_data(struct nvmem_device *nvmem,
|
||||
- struct nvmem_layout *layout)
|
||||
-{
|
||||
- return NULL;
|
||||
-}
|
||||
-
|
||||
#endif /* CONFIG_NVMEM */
|
||||
|
||||
#if IS_ENABLED(CONFIG_NVMEM) && IS_ENABLED(CONFIG_OF)
|
@ -0,0 +1,53 @@
|
||||
From 33cf42e68efc8ff529a7eee08a4f0ba8c8d0a207 Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
|
||||
Date: Thu, 21 Dec 2023 18:34:17 +0100
|
||||
Subject: [PATCH] nvmem: core: add nvmem_dev_size() helper
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
This is required by layouts that need to read whole NVMEM content. It's
|
||||
especially useful for NVMEM devices without hardcoded layout (like
|
||||
U-Boot environment data block).
|
||||
|
||||
Signed-off-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Reviewed-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Link: https://lore.kernel.org/r/20231221173421.13737-2-zajec5@gmail.com
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/core.c | 13 +++++++++++++
|
||||
include/linux/nvmem-consumer.h | 1 +
|
||||
2 files changed, 14 insertions(+)
|
||||
|
||||
--- a/drivers/nvmem/core.c
|
||||
+++ b/drivers/nvmem/core.c
|
||||
@@ -2163,6 +2163,19 @@ const char *nvmem_dev_name(struct nvmem_
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(nvmem_dev_name);
|
||||
|
||||
+/**
|
||||
+ * nvmem_dev_size() - Get the size of a given nvmem device.
|
||||
+ *
|
||||
+ * @nvmem: nvmem device.
|
||||
+ *
|
||||
+ * Return: size of the nvmem device.
|
||||
+ */
|
||||
+size_t nvmem_dev_size(struct nvmem_device *nvmem)
|
||||
+{
|
||||
+ return nvmem->size;
|
||||
+}
|
||||
+EXPORT_SYMBOL_GPL(nvmem_dev_size);
|
||||
+
|
||||
static int __init nvmem_init(void)
|
||||
{
|
||||
int ret;
|
||||
--- a/include/linux/nvmem-consumer.h
|
||||
+++ b/include/linux/nvmem-consumer.h
|
||||
@@ -81,6 +81,7 @@ int nvmem_device_cell_write(struct nvmem
|
||||
struct nvmem_cell_info *info, void *buf);
|
||||
|
||||
const char *nvmem_dev_name(struct nvmem_device *nvmem);
|
||||
+size_t nvmem_dev_size(struct nvmem_device *nvmem);
|
||||
|
||||
void nvmem_add_cell_lookups(struct nvmem_cell_lookup *entries,
|
||||
size_t nentries);
|
@ -0,0 +1,126 @@
|
||||
From 7c8979b42b1a9c5604f431ba804928e55919263c Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
|
||||
Date: Thu, 21 Dec 2023 18:34:18 +0100
|
||||
Subject: [PATCH] nvmem: u-boot-env: use nvmem_add_one_cell() nvmem subsystem
|
||||
helper
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
Simplify adding NVMEM cells.
|
||||
|
||||
Signed-off-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Reviewed-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Link: https://lore.kernel.org/r/20231221173421.13737-3-zajec5@gmail.com
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/u-boot-env.c | 55 +++++++++++++++-----------------------
|
||||
1 file changed, 21 insertions(+), 34 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/u-boot-env.c
|
||||
+++ b/drivers/nvmem/u-boot-env.c
|
||||
@@ -23,13 +23,10 @@ enum u_boot_env_format {
|
||||
|
||||
struct u_boot_env {
|
||||
struct device *dev;
|
||||
+ struct nvmem_device *nvmem;
|
||||
enum u_boot_env_format format;
|
||||
|
||||
struct mtd_info *mtd;
|
||||
-
|
||||
- /* Cells */
|
||||
- struct nvmem_cell_info *cells;
|
||||
- int ncells;
|
||||
};
|
||||
|
||||
struct u_boot_env_image_single {
|
||||
@@ -94,43 +91,36 @@ static int u_boot_env_read_post_process_
|
||||
static int u_boot_env_add_cells(struct u_boot_env *priv, uint8_t *buf,
|
||||
size_t data_offset, size_t data_len)
|
||||
{
|
||||
+ struct nvmem_device *nvmem = priv->nvmem;
|
||||
struct device *dev = priv->dev;
|
||||
char *data = buf + data_offset;
|
||||
char *var, *value, *eq;
|
||||
- int idx;
|
||||
-
|
||||
- priv->ncells = 0;
|
||||
- for (var = data; var < data + data_len && *var; var += strlen(var) + 1)
|
||||
- priv->ncells++;
|
||||
-
|
||||
- priv->cells = devm_kcalloc(dev, priv->ncells, sizeof(*priv->cells), GFP_KERNEL);
|
||||
- if (!priv->cells)
|
||||
- return -ENOMEM;
|
||||
|
||||
- for (var = data, idx = 0;
|
||||
+ for (var = data;
|
||||
var < data + data_len && *var;
|
||||
- var = value + strlen(value) + 1, idx++) {
|
||||
+ var = value + strlen(value) + 1) {
|
||||
+ struct nvmem_cell_info info = {};
|
||||
+
|
||||
eq = strchr(var, '=');
|
||||
if (!eq)
|
||||
break;
|
||||
*eq = '\0';
|
||||
value = eq + 1;
|
||||
|
||||
- priv->cells[idx].name = devm_kstrdup(dev, var, GFP_KERNEL);
|
||||
- if (!priv->cells[idx].name)
|
||||
+ info.name = devm_kstrdup(dev, var, GFP_KERNEL);
|
||||
+ if (!info.name)
|
||||
return -ENOMEM;
|
||||
- priv->cells[idx].offset = data_offset + value - data;
|
||||
- priv->cells[idx].bytes = strlen(value);
|
||||
- priv->cells[idx].np = of_get_child_by_name(dev->of_node, priv->cells[idx].name);
|
||||
+ info.offset = data_offset + value - data;
|
||||
+ info.bytes = strlen(value);
|
||||
+ info.np = of_get_child_by_name(dev->of_node, info.name);
|
||||
if (!strcmp(var, "ethaddr")) {
|
||||
- priv->cells[idx].raw_len = strlen(value);
|
||||
- priv->cells[idx].bytes = ETH_ALEN;
|
||||
- priv->cells[idx].read_post_process = u_boot_env_read_post_process_ethaddr;
|
||||
+ info.raw_len = strlen(value);
|
||||
+ info.bytes = ETH_ALEN;
|
||||
+ info.read_post_process = u_boot_env_read_post_process_ethaddr;
|
||||
}
|
||||
- }
|
||||
|
||||
- if (WARN_ON(idx != priv->ncells))
|
||||
- priv->ncells = idx;
|
||||
+ nvmem_add_one_cell(nvmem, &info);
|
||||
+ }
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -209,7 +199,6 @@ static int u_boot_env_probe(struct platf
|
||||
struct device *dev = &pdev->dev;
|
||||
struct device_node *np = dev->of_node;
|
||||
struct u_boot_env *priv;
|
||||
- int err;
|
||||
|
||||
priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
|
||||
if (!priv)
|
||||
@@ -224,17 +213,15 @@ static int u_boot_env_probe(struct platf
|
||||
return PTR_ERR(priv->mtd);
|
||||
}
|
||||
|
||||
- err = u_boot_env_parse(priv);
|
||||
- if (err)
|
||||
- return err;
|
||||
-
|
||||
config.dev = dev;
|
||||
- config.cells = priv->cells;
|
||||
- config.ncells = priv->ncells;
|
||||
config.priv = priv;
|
||||
config.size = priv->mtd->size;
|
||||
|
||||
- return PTR_ERR_OR_ZERO(devm_nvmem_register(dev, &config));
|
||||
+ priv->nvmem = devm_nvmem_register(dev, &config);
|
||||
+ if (IS_ERR(priv->nvmem))
|
||||
+ return PTR_ERR(priv->nvmem);
|
||||
+
|
||||
+ return u_boot_env_parse(priv);
|
||||
}
|
||||
|
||||
static const struct of_device_id u_boot_env_of_match_table[] = {
|
@ -0,0 +1,81 @@
|
||||
From a832556d23c5a11115f300011a5874d6107a0d62 Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
|
||||
Date: Thu, 21 Dec 2023 18:34:19 +0100
|
||||
Subject: [PATCH] nvmem: u-boot-env: use nvmem device helpers
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
Use nvmem_dev_size() and nvmem_device_read() to make this driver less
|
||||
mtd dependent.
|
||||
|
||||
Signed-off-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Reviewed-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Link: https://lore.kernel.org/r/20231221173421.13737-4-zajec5@gmail.com
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/u-boot-env.c | 23 +++++++++++++++--------
|
||||
1 file changed, 15 insertions(+), 8 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/u-boot-env.c
|
||||
+++ b/drivers/nvmem/u-boot-env.c
|
||||
@@ -127,27 +127,34 @@ static int u_boot_env_add_cells(struct u
|
||||
|
||||
static int u_boot_env_parse(struct u_boot_env *priv)
|
||||
{
|
||||
+ struct nvmem_device *nvmem = priv->nvmem;
|
||||
struct device *dev = priv->dev;
|
||||
size_t crc32_data_offset;
|
||||
size_t crc32_data_len;
|
||||
size_t crc32_offset;
|
||||
size_t data_offset;
|
||||
size_t data_len;
|
||||
+ size_t dev_size;
|
||||
uint32_t crc32;
|
||||
uint32_t calc;
|
||||
- size_t bytes;
|
||||
uint8_t *buf;
|
||||
+ int bytes;
|
||||
int err;
|
||||
|
||||
- buf = kcalloc(1, priv->mtd->size, GFP_KERNEL);
|
||||
+ dev_size = nvmem_dev_size(nvmem);
|
||||
+
|
||||
+ buf = kcalloc(1, dev_size, GFP_KERNEL);
|
||||
if (!buf) {
|
||||
err = -ENOMEM;
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
- err = mtd_read(priv->mtd, 0, priv->mtd->size, &bytes, buf);
|
||||
- if ((err && !mtd_is_bitflip(err)) || bytes != priv->mtd->size) {
|
||||
- dev_err(dev, "Failed to read from mtd: %d\n", err);
|
||||
+ bytes = nvmem_device_read(nvmem, 0, dev_size, buf);
|
||||
+ if (bytes < 0) {
|
||||
+ err = bytes;
|
||||
+ goto err_kfree;
|
||||
+ } else if (bytes != dev_size) {
|
||||
+ err = -EIO;
|
||||
goto err_kfree;
|
||||
}
|
||||
|
||||
@@ -169,8 +176,8 @@ static int u_boot_env_parse(struct u_boo
|
||||
break;
|
||||
}
|
||||
crc32 = le32_to_cpu(*(__le32 *)(buf + crc32_offset));
|
||||
- crc32_data_len = priv->mtd->size - crc32_data_offset;
|
||||
- data_len = priv->mtd->size - data_offset;
|
||||
+ crc32_data_len = dev_size - crc32_data_offset;
|
||||
+ data_len = dev_size - data_offset;
|
||||
|
||||
calc = crc32(~0, buf + crc32_data_offset, crc32_data_len) ^ ~0L;
|
||||
if (calc != crc32) {
|
||||
@@ -179,7 +186,7 @@ static int u_boot_env_parse(struct u_boo
|
||||
goto err_kfree;
|
||||
}
|
||||
|
||||
- buf[priv->mtd->size - 1] = '\0';
|
||||
+ buf[dev_size - 1] = '\0';
|
||||
err = u_boot_env_add_cells(priv, buf, data_offset, data_len);
|
||||
if (err)
|
||||
dev_err(dev, "Failed to add cells: %d\n", err);
|
@ -0,0 +1,62 @@
|
||||
From 6bafe07c930676d6430be471310958070816a595 Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
|
||||
Date: Thu, 21 Dec 2023 18:34:20 +0100
|
||||
Subject: [PATCH] nvmem: u-boot-env: improve coding style
|
||||
MIME-Version: 1.0
|
||||
Content-Type: text/plain; charset=UTF-8
|
||||
Content-Transfer-Encoding: 8bit
|
||||
|
||||
1. Prefer kzalloc() over kcalloc()
|
||||
See memory-allocation.rst which says: "to be on the safe side it's
|
||||
best to use routines that set memory to zero, like kzalloc()"
|
||||
2. Drop dev_err() for u_boot_env_add_cells() fail
|
||||
It can fail only on -ENOMEM. We don't want to print error then.
|
||||
3. Add extra "crc32_addr" variable
|
||||
It makes code reading header's crc32 easier to understand / review.
|
||||
|
||||
Signed-off-by: Rafał Miłecki <rafal@milecki.pl>
|
||||
Reviewed-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
||||
Link: https://lore.kernel.org/r/20231221173421.13737-5-zajec5@gmail.com
|
||||
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
|
||||
---
|
||||
drivers/nvmem/u-boot-env.c | 8 ++++----
|
||||
1 file changed, 4 insertions(+), 4 deletions(-)
|
||||
|
||||
--- a/drivers/nvmem/u-boot-env.c
|
||||
+++ b/drivers/nvmem/u-boot-env.c
|
||||
@@ -132,6 +132,7 @@ static int u_boot_env_parse(struct u_boo
|
||||
size_t crc32_data_offset;
|
||||
size_t crc32_data_len;
|
||||
size_t crc32_offset;
|
||||
+ __le32 *crc32_addr;
|
||||
size_t data_offset;
|
||||
size_t data_len;
|
||||
size_t dev_size;
|
||||
@@ -143,7 +144,7 @@ static int u_boot_env_parse(struct u_boo
|
||||
|
||||
dev_size = nvmem_dev_size(nvmem);
|
||||
|
||||
- buf = kcalloc(1, dev_size, GFP_KERNEL);
|
||||
+ buf = kzalloc(dev_size, GFP_KERNEL);
|
||||
if (!buf) {
|
||||
err = -ENOMEM;
|
||||
goto err_out;
|
||||
@@ -175,7 +176,8 @@ static int u_boot_env_parse(struct u_boo
|
||||
data_offset = offsetof(struct u_boot_env_image_broadcom, data);
|
||||
break;
|
||||
}
|
||||
- crc32 = le32_to_cpu(*(__le32 *)(buf + crc32_offset));
|
||||
+ crc32_addr = (__le32 *)(buf + crc32_offset);
|
||||
+ crc32 = le32_to_cpu(*crc32_addr);
|
||||
crc32_data_len = dev_size - crc32_data_offset;
|
||||
data_len = dev_size - data_offset;
|
||||
|
||||
@@ -188,8 +190,6 @@ static int u_boot_env_parse(struct u_boo
|
||||
|
||||
buf[dev_size - 1] = '\0';
|
||||
err = u_boot_env_add_cells(priv, buf, data_offset, data_len);
|
||||
- if (err)
|
||||
- dev_err(dev, "Failed to add cells: %d\n", err);
|
||||
|
||||
err_kfree:
|
||||
kfree(buf);
|
@ -0,0 +1,93 @@
|
||||
From edd25a77e69b7c546c28077e5dffe72c54c0afe8 Mon Sep 17 00:00:00 2001
|
||||
From: Linus Walleij <linus.walleij@linaro.org>
|
||||
Date: Thu, 21 Sep 2023 22:18:12 +0200
|
||||
Subject: [PATCH 2/4] rtc: rtc7301: Support byte-addressed IO
|
||||
|
||||
The old RTC7301 driver in OpenWrt used byte access, but the
|
||||
current mainline Linux driver uses 32bit word access.
|
||||
|
||||
Make this configurable using device properties using the
|
||||
standard property "reg-io-width" in e.g. device tree.
|
||||
|
||||
This is needed for the USRobotics USR8200 which has the
|
||||
chip connected using byte accesses.
|
||||
|
||||
Debugging and testing by Howard Harte.
|
||||
|
||||
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
|
||||
---
|
||||
drivers/rtc/rtc-r7301.c | 35 +++++++++++++++++++++++++++++++++--
|
||||
1 file changed, 33 insertions(+), 2 deletions(-)
|
||||
|
||||
--- a/drivers/rtc/rtc-r7301.c
|
||||
+++ b/drivers/rtc/rtc-r7301.c
|
||||
@@ -14,6 +14,7 @@
|
||||
#include <linux/module.h>
|
||||
#include <linux/mod_devicetable.h>
|
||||
#include <linux/delay.h>
|
||||
+#include <linux/property.h>
|
||||
#include <linux/regmap.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/rtc.h>
|
||||
@@ -55,12 +56,23 @@ struct rtc7301_priv {
|
||||
u8 bank;
|
||||
};
|
||||
|
||||
-static const struct regmap_config rtc7301_regmap_config = {
|
||||
+/*
|
||||
+ * When the device is memory-mapped, some platforms pack the registers into
|
||||
+ * 32-bit access using the lower 8 bits at each 4-byte stride, while others
|
||||
+ * expose them as simply consecutive bytes.
|
||||
+ */
|
||||
+static const struct regmap_config rtc7301_regmap_32_config = {
|
||||
.reg_bits = 32,
|
||||
.val_bits = 8,
|
||||
.reg_stride = 4,
|
||||
};
|
||||
|
||||
+static const struct regmap_config rtc7301_regmap_8_config = {
|
||||
+ .reg_bits = 8,
|
||||
+ .val_bits = 8,
|
||||
+ .reg_stride = 1,
|
||||
+};
|
||||
+
|
||||
static u8 rtc7301_read(struct rtc7301_priv *priv, unsigned int reg)
|
||||
{
|
||||
int reg_stride = regmap_get_reg_stride(priv->regmap);
|
||||
@@ -356,7 +368,9 @@ static int __init rtc7301_rtc_probe(stru
|
||||
void __iomem *regs;
|
||||
struct rtc7301_priv *priv;
|
||||
struct rtc_device *rtc;
|
||||
+ static const struct regmap_config *mapconf;
|
||||
int ret;
|
||||
+ u32 val;
|
||||
|
||||
priv = devm_kzalloc(&dev->dev, sizeof(*priv), GFP_KERNEL);
|
||||
if (!priv)
|
||||
@@ -366,8 +380,25 @@ static int __init rtc7301_rtc_probe(stru
|
||||
if (IS_ERR(regs))
|
||||
return PTR_ERR(regs);
|
||||
|
||||
+ ret = device_property_read_u32(&dev->dev, "reg-io-width", &val);
|
||||
+ if (ret)
|
||||
+ /* Default to 32bit accesses */
|
||||
+ val = 4;
|
||||
+
|
||||
+ switch (val) {
|
||||
+ case 1:
|
||||
+ mapconf = &rtc7301_regmap_8_config;
|
||||
+ break;
|
||||
+ case 4:
|
||||
+ mapconf = &rtc7301_regmap_32_config;
|
||||
+ break;
|
||||
+ default:
|
||||
+ dev_err(&dev->dev, "invalid reg-io-width %d\n", val);
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
priv->regmap = devm_regmap_init_mmio(&dev->dev, regs,
|
||||
- &rtc7301_regmap_config);
|
||||
+ mapconf);
|
||||
if (IS_ERR(priv->regmap))
|
||||
return PTR_ERR(priv->regmap);
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user