Received: by 2002:ad5:4acb:0:0:0:0:0 with SMTP id n11csp940096imw; Wed, 13 Jul 2022 10:40:45 -0700 (PDT) X-Google-Smtp-Source: AGRyM1viYWiRYmB8ZdqjpO2XVdVmQ5TPokIaEATUhbgW8jVd/2nRbrg+zZtXGAvY29PNRwxocLsA X-Received: by 2002:a05:6402:5114:b0:43a:d072:83b9 with SMTP id m20-20020a056402511400b0043ad07283b9mr6684684edd.260.1657734045002; Wed, 13 Jul 2022 10:40:45 -0700 (PDT) ARC-Seal: i=2; a=rsa-sha256; t=1657734044; cv=pass; d=google.com; s=arc-20160816; b=KvOsFW+iO9uZHSfv5Ps4s2zBYNzrfZvpETSMAkpF8unrf9XS/2hxI15Hd71+4AlKsu sBJlm5JtXUvPKQQwXljHtI+X0Xppg620+JEi49VMgBafmcdWauHLYXreAc7UDYUT5fCU +wG4ZqECT6JfEw1W567iHeidIk/uqEndj6pFI3wcO3r1QfJw2FTj0DpoTEPReXWmH9Ag Mc3vLeufYPcnjlRoYk0IUo3I84NJmTdgB4w9t1E7vSxJ9jgU5LgsGCBXdJ9R3k081EHH /y212R71Y3vD1NqnAh7rqbB6cUneEvsuEOzM4tccjczu82WEXrg/9VOVyjcpQgmAVnqC zc3Q== ARC-Message-Signature: i=2; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:mime-version:message-id:date:subject:cc:to:from :dkim-signature; bh=YgLlpO3crQVVXxpQN81awP9BMX90WD2zadYZzchvlaM=; b=j3cWqr2W8O1sCPqHzf3nCaWyZ6CzyJrOTeQbWU2ZanCPHBe51Ezv4HwUL4G89M91IE 9R1ratofJDnWl/gl4C41H7nZzPi6LqwsWeS9BH3y+/fgFsGvS0bA76S84MxI5zORmOw3 WvvnNLYI8aEKDropwMitMqG91RhcCNDYHvK5IZIiYcXefqFK5gW9/DkKzPH1Tn+MhkC3 mhP7eM9t/jptua+YuqRhpl0cW6X7frF052VqqyMYcydAD6nIVLB1STsjYg1fh5JhI7Mf 9waGOA5UcLct+Zu61OKcxIzNpaLqiUUAL3Dtj67eIAR96LI/b9w9H8tVBBW2GZGFoTPc zVoA== ARC-Authentication-Results: i=2; mx.google.com; dkim=pass header.i=@plvision.eu header.s=selector2 header.b=KwWb01Mt; arc=pass (i=1 spf=pass spfdomain=plvision.eu dkim=pass dkdomain=plvision.eu dmarc=pass fromdomain=plvision.eu); spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=plvision.eu Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id q37-20020a05640224a500b0043a6ddb2941si16459798eda.253.2022.07.13.10.40.18; Wed, 13 Jul 2022 10:40:44 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@plvision.eu header.s=selector2 header.b=KwWb01Mt; arc=pass (i=1 spf=pass spfdomain=plvision.eu dkim=pass dkdomain=plvision.eu dmarc=pass fromdomain=plvision.eu); spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=plvision.eu Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231494AbiGMRUu (ORCPT + 99 others); Wed, 13 Jul 2022 13:20:50 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:47252 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229861AbiGMRUs (ORCPT ); Wed, 13 Jul 2022 13:20:48 -0400 Received: from EUR03-AM5-obe.outbound.protection.outlook.com (mail-eopbgr30107.outbound.protection.outlook.com [40.107.3.107]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 2050DE2; Wed, 13 Jul 2022 10:20:46 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=iHqGRQ46oOY9BNrjKxpMY0RDfimJJ5Aihnb3B2s5IghnqA4hyViFjbnvYX5ecnf3YlwXKogpmiI+pHjUWNTzNHTCdkdYVkkSHOfylfbW4S8ki+8+C1lcRzYnfchU2KfTUrLzuA5GFbolH2zNqTEfgh6UskeLw0S1yD39V2vCNhgSU4V+Ip8zkBHPw5li2kj85Cf2CapG6lZssgh75RXlXhegcZauU1PTUE6JIAV47xY0Dxr7W42AnQoPyS7V531Qx3ftQODIHrR/G12NEyyIXwh1l3g53piUA6E6ZF+ER8S3n1B0876E3Id52sW+3csDARrWpnR0PePjqkuyF9PrMw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=arcselector9901; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-AntiSpam-MessageData-ChunkCount:X-MS-Exchange-AntiSpam-MessageData-0:X-MS-Exchange-AntiSpam-MessageData-1; bh=YgLlpO3crQVVXxpQN81awP9BMX90WD2zadYZzchvlaM=; b=WtN3e6PVFB+6X35ChDCufBfMtOsy267O5bQGhAO9qFXy/ISOxIQ+ISjgbj+ohJfZi775wr/1Otji1S3spRA5qC7AS4xv8+zMdz4O2J6T+EkJvxBBS1oYmzdo8gfEAfYNIJOWiHwDjddYqM7fw5CUSRjnbd7fuePad782LoHijIo6gSg/7n3u/mEFky6s/od+TILM3ghXwavI90h0YLol0jADu2eISSCsT0Ca+Lg9zesJ28rVZeFY5KzTauIc1hQhEska4vsx6uHj35EwCP0viKLbGBjbXszet5Dfe+YoEwM/F5LzikYgzZRoX+glGjRPcl4NzhY64b18q3hk+eC/PQ== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=pass smtp.mailfrom=plvision.eu; dmarc=pass action=none header.from=plvision.eu; dkim=pass header.d=plvision.eu; arc=none DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=plvision.eu; s=selector2; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=YgLlpO3crQVVXxpQN81awP9BMX90WD2zadYZzchvlaM=; b=KwWb01Mtm06zfbuMnT/nz2OxqbAZxwmf/pEpV+A0yiEelnFk/kqbVAX9b/dqu7g/ZYeJ5+pa+IuiuasTL9G0XfdJL8truM/ePgk6VSOnhbeojLjeYZSAROkGebODGOLI6yXSam/KdQqkwqO4u0ihh5626Z290ueAqbxINTiaMiU= Authentication-Results: dkim=none (message not signed) header.d=none;dmarc=none action=none header.from=plvision.eu; Received: from GV1P190MB2019.EURP190.PROD.OUTLOOK.COM (2603:10a6:150:5b::20) by GV1P190MB1946.EURP190.PROD.OUTLOOK.COM (2603:10a6:150:55::16) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.5438.12; Wed, 13 Jul 2022 17:20:42 +0000 Received: from GV1P190MB2019.EURP190.PROD.OUTLOOK.COM ([fe80::25a3:3edc:9320:f35e]) by GV1P190MB2019.EURP190.PROD.OUTLOOK.COM ([fe80::25a3:3edc:9320:f35e%3]) with mapi id 15.20.5417.023; Wed, 13 Jul 2022 17:20:41 +0000 From: Oleksandr Mazur To: netdev@vger.kernel.org Cc: linux-kernel@vger.kernel.org, oleksandr.mazur@plvision.eu, davem@davemloft.net, edumazet@google.com, kuba@kernel.org, pabeni@redhat.com, yevhen.orlov@plvision.eu, taras.chornyi@plvision.eu, linux@armlinux.org.uk Subject: [PATCH V2 net-next] net: marvell: prestera: add phylink support Date: Wed, 13 Jul 2022 20:20:13 +0300 Message-Id: <20220713172013.29531-1-oleksandr.mazur@plvision.eu> X-Mailer: git-send-email 2.17.1 Content-Type: text/plain X-ClientProxiedBy: GV3P280CA0065.SWEP280.PROD.OUTLOOK.COM (2603:10a6:150:a::23) To GV1P190MB2019.EURP190.PROD.OUTLOOK.COM (2603:10a6:150:5b::20) MIME-Version: 1.0 X-MS-PublicTrafficType: Email X-MS-Office365-Filtering-Correlation-Id: a0c58748-e993-4f65-de07-08da64f40326 X-MS-TrafficTypeDiagnostic: GV1P190MB1946:EE_ X-MS-Exchange-SenderADCheck: 1 X-MS-Exchange-AntiSpam-Relay: 0 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: FZ/1fRCo+hAHjm7hOXBpixoEHPovAi8r88vHJFW4Iva21g2h60pgQXqHGkMhPzLn+MrOfX+bNtDqhtd9MM5ys7SgJtxSSKs8VIReqTS0O2r0/nJ6kGgSpTCGeU/XX5HKhBK1o/OwoOF3tpWc4zNM0wUt21uCyQdjloCDhpRlGK5JJFMyNOXrNfT3nKWal5crYeC2KpWsyoHCZXok9jrXjCoD84OQYFbqjKQwIm2CW4+zLE0+HYCidMb10hAPdlzx+LynGmMqaVs6AfBQJvFGWU+P0+bQoFQ8FsZkFeMIxEbTjfdf56b7o291Y0SBkWTptXpC5u+bemn0yB7DD9OU864O3FPj3V4TWWz36pUNXYUkDCgyoycXPYyOrmJU6HyAYVn+e1eozuBoEVhC+fzlcO9/YmBuHGizPyL/DNHEjQyvT8kEe7LSFzWDsh6pu7C++pghHanwb68T4jGYr3afzmhmXeV7Uiru/ZfxLr30NjkMJ7t/DyUC+8j1Jw27T9ZlOCxu58PWYPUk/pKVDxw/G5ITtZE5cF+oK1NljPliSKwuMrRZSah//ybA1VuMJCF7A4HN9+WmcoGK38iMUx8ZiI3sUoke0D0MoajcRByz+js4uOl1WitsJNZR0GZbraCfQ0Z79qya9EKyONApXPi4a+hdORUlbrifhqmhWWP5F7MDkXC1HeJCBgf0mpwLdRmGrJSNMl8J8WDQlfisxdl0tlGsCOWMEJhgbG0bn95/YbxEHSFtL58rVwTU2xj+EM2+n3glXrlpGAJCXjU7s5wH8NSWLdAMZ/btzRvEJxs97J2r7WyNHFute5PMvy1Du/Z1 X-Forefront-Antispam-Report: CIP:255.255.255.255;CTRY:;LANG:en;SCL:1;SRV:;IPV:NLI;SFV:NSPM;H:GV1P190MB2019.EURP190.PROD.OUTLOOK.COM;PTR:;CAT:NONE;SFS:(13230016)(4636009)(396003)(136003)(39830400003)(346002)(366004)(376002)(6916009)(5660300002)(66946007)(1076003)(86362001)(36756003)(44832011)(4326008)(8676002)(66556008)(8936002)(316002)(66476007)(38350700002)(41300700001)(2616005)(6512007)(26005)(6666004)(6506007)(52116002)(38100700002)(478600001)(6486002)(2906002)(66574015)(186003)(83380400001)(30864003);DIR:OUT;SFP:1102; X-MS-Exchange-AntiSpam-MessageData-ChunkCount: 1 X-MS-Exchange-AntiSpam-MessageData-0: =?us-ascii?Q?CElrYu7D7TyGaGtETEcMEhMC2Lsx/7V6RWx5UcY1athZnyjyPC2z6Qhcc7sx?= =?us-ascii?Q?2jRx6rAJYwAKXA/Bc3dHS1UOAk/AGBBxs4ndnzbBJCoxezrl4WxE2c3NPFlg?= =?us-ascii?Q?Ib1wJVFvgSD0HQ+Y2vr1eJlL74USjjFfR0P3quJWUF2O/0zZi6X+Kymgy0AK?= =?us-ascii?Q?nRcR9x28tpcNvhpNyQlQjJUwI/cvcwVC8H+ByBUpz8e72KNHi6AVzca8wE9H?= =?us-ascii?Q?fZH4ToSwL13QosphorViXNVmNHUBgNFcy4llzM0ueJsDKzKboCwkllkc4AnS?= =?us-ascii?Q?7zHXAtBMqR+sQtnDVR2eIwjwdYV2x2OfH6iGGAOaLGZqmkpZiqvJj1aDbmIm?= =?us-ascii?Q?ptk6uQnTHYRlJ24wPnJ+rCRu/Tzy+3peHftt+qflFstxIc7RNj1A31J1PkKy?= =?us-ascii?Q?UADlNt/zbEcbnpEDTKQ59cKBmXbVtNUdPiftklPAe6yGq4kdMhhrabA6yqrW?= =?us-ascii?Q?tqrRqznHr0Qk7Sl59MtyeoTiT4XOZMDQwI5kBgrHNjmZ8Lb1v55K05HwBvwZ?= =?us-ascii?Q?6dHr49ecrRP9G45L6AMlIz0s0uJydPSe9GWSv1Ffin47QXSZanPl2DYHLP4J?= =?us-ascii?Q?e4GjzvazMLY2JD5ilvQ14qD81XmTbM5Jjt0YF3IC+GKhRZAovDMyJPclfPC4?= =?us-ascii?Q?7lHInEFNMxa8JTXR9yhG/tVAWCPKwJOAVZvjI8YPH0+1zh+/x8p+Phlx2XCb?= =?us-ascii?Q?YoNdIllzv8fN3rTqxzdDqCkjyEdgViPIqHCAG2N4JuXhjD2eGV4JxF4Y1OI3?= =?us-ascii?Q?cM8yn7unRoUv1xeMujXQ265OnQRYLbWlLw7BVxsC9Qv1K6vfsHxcmX7CZ7s9?= =?us-ascii?Q?EVLhlsYl/eV1+NkCz/WJzN6w7FdSAr9lZ+LF02XM1DSwT6aydP0fCDwST3tf?= =?us-ascii?Q?vG4fsMgE1rORtifG0UQUfUi8ahMI3y/kZ31otT3XOLBcRJEHOBZISv+FCfEl?= =?us-ascii?Q?AwTzJciQD1X2jKD+kR/5wE9CAk8j3Kt/zlwSmwcuQdiisSSN0xZGDXq8ayF0?= =?us-ascii?Q?k5WwHwhxy2V0+zbCHDXFzbMorxIQZBH70IxSNa25DCqt9yBeEFxxGt45AyKq?= =?us-ascii?Q?Tv2JFbYfKM0R7pGXqCXB7djBZWeu7XWx7QLl/e0kKILyEpQixFDeecpp5ByX?= =?us-ascii?Q?9k4jqFQm9T29h9XbqoNCeXXGrCnD2l/+VULSxQl3+Mfs7OXBtbw2FXRZ54Zu?= =?us-ascii?Q?VkRmt7z6Htppn0WGPlirt7rS0XDPp/HmciqiJjLmGGr/oXUdO/OvqQzsG/dt?= =?us-ascii?Q?aikMmQWwwcawmvKECQb9iXoLM2rd5nOUP8DyMiFm5WZos+8oOXMlpL2hhYxx?= =?us-ascii?Q?mjdTnWowy6ykLNu6Y4s/YrLItPidDaXOVnT/jIQjTv7Dq/0G+HBDcwxbuBrx?= =?us-ascii?Q?ApNX/ILRj/OOYXyDFD0ddMcw3rEmBnDHrMidA1woptYK3Em6U0ZPfn5Wx2Pv?= =?us-ascii?Q?7922vbR7yjbHvRKTLbTAXVZ7RiJ7Wcf/WB4yRurWn3DlJYFqCdTlxajtjlE4?= =?us-ascii?Q?a/6qcGyEqjlqffz0EFXW01DhT74HdvJ68YqeDU5eLOhuvf4cY/MwBYP2jaVz?= =?us-ascii?Q?+2SJMUpYUjpZbmYEcvcOlA70eC40Yb3LngrzeCUHTRnlxPoMeNcUcNSQiOac?= =?us-ascii?Q?2Q=3D=3D?= X-OriginatorOrg: plvision.eu X-MS-Exchange-CrossTenant-Network-Message-Id: a0c58748-e993-4f65-de07-08da64f40326 X-MS-Exchange-CrossTenant-AuthSource: GV1P190MB2019.EURP190.PROD.OUTLOOK.COM X-MS-Exchange-CrossTenant-AuthAs: Internal X-MS-Exchange-CrossTenant-OriginalArrivalTime: 13 Jul 2022 17:20:41.6678 (UTC) X-MS-Exchange-CrossTenant-FromEntityHeader: Hosted X-MS-Exchange-CrossTenant-Id: 03707b74-30f3-46b6-a0e0-ff0a7438c9c4 X-MS-Exchange-CrossTenant-MailboxType: HOSTED X-MS-Exchange-CrossTenant-UserPrincipalName: jYyrK5J7VGH9sl38yav0AojYu0UbVw6ybN+qZchYWasMl6tBaP0HdoybXjzNOjc7fiQhskrfdrWzuKoVpr600Q0AH2Gqst7EtyMIcFYFaTI= X-MS-Exchange-Transport-CrossTenantHeadersStamped: GV1P190MB1946 X-Spam-Status: No, score=-2.1 required=5.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,RCVD_IN_DNSWL_NONE, RCVD_IN_MSPIKE_H2,SPF_HELO_PASS,SPF_PASS,T_SCC_BODY_TEXT_LINE autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org For SFP port prestera driver will use kernel phylink infrastucture to configure port mode based on the module that has beed inserted Co-developed-by: Yevhen Orlov Signed-off-by: Yevhen Orlov Co-developed-by: Taras Chornyi Signed-off-by: Taras Chornyi Signed-off-by: Oleksandr Mazur PATCH V2: - fix mistreat of bitfield values as if they were bools. - remove phylink_config ifdefs. - remove obsolete phylink pcs / mac callbacks; - rework mac (/pcs) config to not look for speed / duplex parameters while link is not yet set up. - remove unused functions. - add phylink select cfg to prestera Kconfig. --- drivers/net/ethernet/marvell/prestera/Kconfig | 1 + .../net/ethernet/marvell/prestera/prestera.h | 9 + .../marvell/prestera/prestera_ethtool.c | 28 +- .../marvell/prestera/prestera_ethtool.h | 3 - .../ethernet/marvell/prestera/prestera_main.c | 348 ++++++++++++++++-- 5 files changed, 327 insertions(+), 62 deletions(-) diff --git a/drivers/net/ethernet/marvell/prestera/Kconfig b/drivers/net/ethernet/marvell/prestera/Kconfig index b6f20e2034c6..f2f7663c3d10 100644 --- a/drivers/net/ethernet/marvell/prestera/Kconfig +++ b/drivers/net/ethernet/marvell/prestera/Kconfig @@ -8,6 +8,7 @@ config PRESTERA depends on NET_SWITCHDEV && VLAN_8021Q depends on BRIDGE || BRIDGE=n select NET_DEVLINK + select PHYLINK help This driver supports Marvell Prestera Switch ASICs family. diff --git a/drivers/net/ethernet/marvell/prestera/prestera.h b/drivers/net/ethernet/marvell/prestera/prestera.h index bff9651f0a89..832c27e0c284 100644 --- a/drivers/net/ethernet/marvell/prestera/prestera.h +++ b/drivers/net/ethernet/marvell/prestera/prestera.h @@ -7,6 +7,7 @@ #include #include #include +#include #include #include @@ -92,6 +93,7 @@ struct prestera_lag { struct prestera_flow_block; struct prestera_port_mac_state { + bool valid; u32 mode; u32 speed; bool oper; @@ -151,6 +153,12 @@ struct prestera_port { struct prestera_port_phy_config cfg_phy; struct prestera_port_mac_state state_mac; struct prestera_port_phy_state state_phy; + + struct phylink_config phy_config; + struct phylink *phy_link; + struct phylink_pcs phylink_pcs; + + rwlock_t state_mac_lock; }; struct prestera_device { @@ -291,6 +299,7 @@ struct prestera_switch { u32 mtu_min; u32 mtu_max; u8 id; + struct device_node *np; struct prestera_router *router; struct prestera_lag *lags; struct prestera_counter *counter; diff --git a/drivers/net/ethernet/marvell/prestera/prestera_ethtool.c b/drivers/net/ethernet/marvell/prestera/prestera_ethtool.c index 40d5b89573bb..1da7ff889417 100644 --- a/drivers/net/ethernet/marvell/prestera/prestera_ethtool.c +++ b/drivers/net/ethernet/marvell/prestera/prestera_ethtool.c @@ -521,6 +521,9 @@ prestera_ethtool_get_link_ksettings(struct net_device *dev, ecmd->base.speed = SPEED_UNKNOWN; ecmd->base.duplex = DUPLEX_UNKNOWN; + if (port->phy_link) + return phylink_ethtool_ksettings_get(port->phy_link, ecmd); + ecmd->base.autoneg = port->autoneg ? AUTONEG_ENABLE : AUTONEG_DISABLE; if (port->caps.type == PRESTERA_PORT_TYPE_TP) { @@ -648,6 +651,9 @@ prestera_ethtool_set_link_ksettings(struct net_device *dev, u8 adver_fec; int err; + if (port->phy_link) + return phylink_ethtool_ksettings_set(port->phy_link, ecmd); + err = prestera_port_type_set(ecmd, port); if (err) return err; @@ -782,28 +788,6 @@ static int prestera_ethtool_nway_reset(struct net_device *dev) return -EINVAL; } -void prestera_ethtool_port_state_changed(struct prestera_port *port, - struct prestera_port_event *evt) -{ - struct prestera_port_mac_state *smac = &port->state_mac; - - smac->oper = evt->data.mac.oper; - - if (smac->oper) { - smac->mode = evt->data.mac.mode; - smac->speed = evt->data.mac.speed; - smac->duplex = evt->data.mac.duplex; - smac->fc = evt->data.mac.fc; - smac->fec = evt->data.mac.fec; - } else { - smac->mode = PRESTERA_MAC_MODE_MAX; - smac->speed = SPEED_UNKNOWN; - smac->duplex = DUPLEX_UNKNOWN; - smac->fc = 0; - smac->fec = 0; - } -} - const struct ethtool_ops prestera_ethtool_ops = { .get_drvinfo = prestera_ethtool_get_drvinfo, .get_link_ksettings = prestera_ethtool_get_link_ksettings, diff --git a/drivers/net/ethernet/marvell/prestera/prestera_ethtool.h b/drivers/net/ethernet/marvell/prestera/prestera_ethtool.h index 9eb18e99dea6..bd5600886bc6 100644 --- a/drivers/net/ethernet/marvell/prestera/prestera_ethtool.h +++ b/drivers/net/ethernet/marvell/prestera/prestera_ethtool.h @@ -11,7 +11,4 @@ struct prestera_port; extern const struct ethtool_ops prestera_ethtool_ops; -void prestera_ethtool_port_state_changed(struct prestera_port *port, - struct prestera_port_event *evt); - #endif /* _PRESTERA_ETHTOOL_H_ */ diff --git a/drivers/net/ethernet/marvell/prestera/prestera_main.c b/drivers/net/ethernet/marvell/prestera/prestera_main.c index ea5bd5069826..a0d9c186df6b 100644 --- a/drivers/net/ethernet/marvell/prestera/prestera_main.c +++ b/drivers/net/ethernet/marvell/prestera/prestera_main.c @@ -9,6 +9,7 @@ #include #include #include +#include #include "prestera.h" #include "prestera_hw.h" @@ -142,18 +143,24 @@ static int prestera_port_open(struct net_device *dev) struct prestera_port_mac_config cfg_mac; int err = 0; - if (port->caps.transceiver == PRESTERA_PORT_TCVR_SFP) { - err = prestera_port_cfg_mac_read(port, &cfg_mac); - if (!err) { - cfg_mac.admin = true; - err = prestera_port_cfg_mac_write(port, &cfg_mac); - } + if (port->phy_link) { + phylink_start(port->phy_link); } else { - port->cfg_phy.admin = true; - err = prestera_hw_port_phy_mode_set(port, true, port->autoneg, - port->cfg_phy.mode, - port->adver_link_modes, - port->cfg_phy.mdix); + if (port->caps.transceiver == PRESTERA_PORT_TCVR_SFP) { + err = prestera_port_cfg_mac_read(port, &cfg_mac); + if (!err) { + cfg_mac.admin = true; + err = prestera_port_cfg_mac_write(port, + &cfg_mac); + } + } else { + port->cfg_phy.admin = true; + err = prestera_hw_port_phy_mode_set(port, true, + port->autoneg, + port->cfg_phy.mode, + port->adver_link_modes, + port->cfg_phy.mdix); + } } netif_start_queue(dev); @@ -169,23 +176,254 @@ static int prestera_port_close(struct net_device *dev) netif_stop_queue(dev); - if (port->caps.transceiver == PRESTERA_PORT_TCVR_SFP) { + if (port->phy_link) { + phylink_stop(port->phy_link); + phylink_disconnect_phy(port->phy_link); err = prestera_port_cfg_mac_read(port, &cfg_mac); if (!err) { cfg_mac.admin = false; prestera_port_cfg_mac_write(port, &cfg_mac); } } else { - port->cfg_phy.admin = false; - err = prestera_hw_port_phy_mode_set(port, false, port->autoneg, - port->cfg_phy.mode, - port->adver_link_modes, - port->cfg_phy.mdix); + if (port->caps.transceiver == PRESTERA_PORT_TCVR_SFP) { + err = prestera_port_cfg_mac_read(port, &cfg_mac); + if (!err) { + cfg_mac.admin = false; + prestera_port_cfg_mac_write(port, &cfg_mac); + } + } else { + port->cfg_phy.admin = false; + err = prestera_hw_port_phy_mode_set(port, false, port->autoneg, + port->cfg_phy.mode, + port->adver_link_modes, + port->cfg_phy.mdix); + } + } + + return err; +} + +static void +prestera_port_mac_state_cache_read(struct prestera_port *port, + struct prestera_port_mac_state *state) +{ + read_lock(&port->state_mac_lock); + *state = port->state_mac; + read_unlock(&port->state_mac_lock); +} + +static void +prestera_port_mac_state_cache_write(struct prestera_port *port, + struct prestera_port_mac_state *state) +{ + write_lock(&port->state_mac_lock); + port->state_mac = *state; + write_unlock(&port->state_mac_lock); +} + +static struct prestera_port *prestera_pcs_to_port(struct phylink_pcs *pcs) +{ + return container_of(pcs, struct prestera_port, phylink_pcs); +} + +static void prestera_mac_config(struct phylink_config *config, + unsigned int an_mode, + const struct phylink_link_state *state) +{ +} + +static void prestera_mac_link_down(struct phylink_config *config, + unsigned int mode, phy_interface_t interface) +{ + struct net_device *ndev = to_net_dev(config->dev); + struct prestera_port *port = netdev_priv(ndev); + struct prestera_port_mac_state state_mac; + + /* Invalidate. Parameters will update on next link event. */ + memset(&state_mac, 0, sizeof(state_mac)); + state_mac.valid = false; + prestera_port_mac_state_cache_write(port, &state_mac); +} + +static void prestera_mac_link_up(struct phylink_config *config, + struct phy_device *phy, + unsigned int mode, phy_interface_t interface, + int speed, int duplex, + bool tx_pause, bool rx_pause) +{ +} + +static struct phylink_pcs *prestera_mac_select_pcs(struct phylink_config *config, + phy_interface_t interface) +{ + struct net_device *dev = to_net_dev(config->dev); + struct prestera_port *port = netdev_priv(dev); + + return &port->phylink_pcs; +} + +static void prestera_pcs_get_state(struct phylink_pcs *pcs, + struct phylink_link_state *state) +{ + struct prestera_port *port = container_of(pcs, struct prestera_port, + phylink_pcs); + struct prestera_port_mac_state smac; + + prestera_port_mac_state_cache_read(port, &smac); + + if (smac.valid) { + state->link = smac.oper ? 1 : 0; + /* AN is completed, when port is up */ + state->an_complete = (smac.oper && port->autoneg) ? 1 : 0; + state->speed = smac.speed; + state->duplex = smac.duplex; + } else { + state->link = 0; + state->an_complete = 0; + } +} + +static int prestera_pcs_config(struct phylink_pcs *pcs, + unsigned int mode, + phy_interface_t interface, + const unsigned long *advertising, + bool permit_pause_to_mac) +{ + struct prestera_port *port = port = prestera_pcs_to_port(pcs); + struct prestera_port_mac_config cfg_mac; + int err; + + err = prestera_port_cfg_mac_read(port, &cfg_mac); + if (err) + return err; + + cfg_mac.admin = true; + cfg_mac.fec = PRESTERA_PORT_FEC_OFF; + + switch (interface) { + case PHY_INTERFACE_MODE_10GBASER: + cfg_mac.speed = SPEED_10000; + cfg_mac.inband = 0; + cfg_mac.mode = PRESTERA_MAC_MODE_SR_LR; + break; + case PHY_INTERFACE_MODE_2500BASEX: + cfg_mac.speed = SPEED_2500; + cfg_mac.duplex = DUPLEX_FULL; + cfg_mac.inband = test_bit(ETHTOOL_LINK_MODE_Autoneg_BIT, + advertising); + cfg_mac.mode = PRESTERA_MAC_MODE_SGMII; + break; + case PHY_INTERFACE_MODE_SGMII: + cfg_mac.inband = test_bit(ETHTOOL_LINK_MODE_Autoneg_BIT, + advertising); + cfg_mac.mode = PRESTERA_MAC_MODE_SGMII; + break; + case PHY_INTERFACE_MODE_1000BASEX: + default: + cfg_mac.speed = SPEED_1000; + cfg_mac.duplex = DUPLEX_FULL; + cfg_mac.inband = test_bit(ETHTOOL_LINK_MODE_Autoneg_BIT, + advertising); + cfg_mac.mode = PRESTERA_MAC_MODE_1000BASE_X; + break; } + err = prestera_port_cfg_mac_write(port, &cfg_mac); + if (err) + return err; + + return 0; +} + +static void prestera_pcs_an_restart(struct phylink_pcs *pcs) +{ +} + +static const struct phylink_mac_ops prestera_mac_ops = { + .validate = phylink_generic_validate, + .mac_select_pcs = prestera_mac_select_pcs, + .mac_config = prestera_mac_config, + .mac_link_down = prestera_mac_link_down, + .mac_link_up = prestera_mac_link_up, +}; + +static const struct phylink_pcs_ops prestera_pcs_ops = { + .pcs_get_state = prestera_pcs_get_state, + .pcs_config = prestera_pcs_config, + .pcs_an_restart = prestera_pcs_an_restart, +}; + +static int prestera_port_sfp_bind(struct prestera_port *port) +{ + struct prestera_switch *sw = port->sw; + struct device_node *ports, *node; + struct fwnode_handle *fwnode; + struct phylink *phy_link; + int err; + + if (!sw->np) + return 0; + + ports = of_find_node_by_name(sw->np, "ports"); + + for_each_child_of_node(ports, node) { + int num; + + err = of_property_read_u32(node, "prestera,port-num", &num); + if (err) { + dev_err(sw->dev->dev, + "device node %pOF has no valid reg property: %d\n", + node, err); + goto out; + } + + if (port->fp_id != num) + continue; + + port->phylink_pcs.ops = &prestera_pcs_ops; + + port->phy_config.dev = &port->dev->dev; + port->phy_config.type = PHYLINK_NETDEV; + + fwnode = of_fwnode_handle(node); + + __set_bit(PHY_INTERFACE_MODE_10GBASER, + port->phy_config.supported_interfaces); + __set_bit(PHY_INTERFACE_MODE_2500BASEX, + port->phy_config.supported_interfaces); + __set_bit(PHY_INTERFACE_MODE_SGMII, + port->phy_config.supported_interfaces); + __set_bit(PHY_INTERFACE_MODE_1000BASEX, + port->phy_config.supported_interfaces); + + port->phy_config.mac_capabilities = MAC_1000 | MAC_2500FD | MAC_10000FD; + + phy_link = phylink_create(&port->phy_config, fwnode, + PHY_INTERFACE_MODE_INTERNAL, + &prestera_mac_ops); + if (IS_ERR(phy_link)) { + netdev_err(port->dev, "failed to create phylink\n"); + err = PTR_ERR(phy_link); + goto out; + } + + port->phy_link = phy_link; + break; + } + +out: + of_node_put(ports); return err; } +static int prestera_port_sfp_unbind(struct prestera_port *port) +{ + if (port->phy_link) + phylink_destroy(port->phy_link); + + return 0; +} + static netdev_tx_t prestera_port_xmit(struct sk_buff *skb, struct net_device *dev) { @@ -380,8 +618,10 @@ static int prestera_port_create(struct prestera_switch *sw, u32 id) dev->features |= NETIF_F_NETNS_LOCAL | NETIF_F_HW_TC; dev->netdev_ops = &prestera_netdev_ops; dev->ethtool_ops = &prestera_ethtool_ops; + SET_NETDEV_DEV(dev, sw->dev->dev); - netif_carrier_off(dev); + if (port->caps.transceiver != PRESTERA_PORT_TCVR_SFP) + netif_carrier_off(dev); dev->mtu = min_t(unsigned int, sw->mtu_max, PRESTERA_MTU_DEFAULT); dev->min_mtu = sw->mtu_min; @@ -432,7 +672,7 @@ static int prestera_port_create(struct prestera_switch *sw, u32 id) cfg_mac.admin = false; cfg_mac.mode = PRESTERA_MAC_MODE_MAX; } - cfg_mac.inband = false; + cfg_mac.inband = 0; cfg_mac.speed = 0; cfg_mac.duplex = DUPLEX_UNKNOWN; cfg_mac.fec = PRESTERA_PORT_FEC_OFF; @@ -474,8 +714,13 @@ static int prestera_port_create(struct prestera_switch *sw, u32 id) prestera_devlink_port_set(port); + err = prestera_port_sfp_bind(port); + if (err) + goto err_sfp_bind; + return 0; +err_sfp_bind: err_register_netdev: prestera_port_list_del(port); err_port_init: @@ -521,8 +766,10 @@ static int prestera_create_ports(struct prestera_switch *sw) return 0; err_port_create: - list_for_each_entry_safe(port, tmp, &sw->port_list, list) + list_for_each_entry_safe(port, tmp, &sw->port_list, list) { + prestera_port_sfp_unbind(port); prestera_port_destroy(port); + } return err; } @@ -530,25 +777,48 @@ static int prestera_create_ports(struct prestera_switch *sw) static void prestera_port_handle_event(struct prestera_switch *sw, struct prestera_event *evt, void *arg) { + struct prestera_port_mac_state smac; + struct prestera_port_event *pevt; struct delayed_work *caching_dw; struct prestera_port *port; - port = prestera_find_port(sw, evt->port_evt.port_id); - if (!port || !port->dev) - return; - - caching_dw = &port->cached_hw_stats.caching_dw; - - prestera_ethtool_port_state_changed(port, &evt->port_evt); - if (evt->id == PRESTERA_PORT_EVENT_MAC_STATE_CHANGED) { + pevt = &evt->port_evt; + port = prestera_find_port(sw, pevt->port_id); + if (!port || !port->dev) + return; + + caching_dw = &port->cached_hw_stats.caching_dw; + + if (port->phy_link) { + memset(&smac, 0, sizeof(smac)); + smac.valid = true; + smac.oper = pevt->data.mac.oper; + if (smac.oper) { + smac.mode = pevt->data.mac.mode; + smac.speed = pevt->data.mac.speed; + smac.duplex = pevt->data.mac.duplex; + smac.fc = pevt->data.mac.fc; + smac.fec = pevt->data.mac.fec; + } + prestera_port_mac_state_cache_write(port, &smac); + } + if (port->state_mac.oper) { - netif_carrier_on(port->dev); + if (port->phy_link) + phylink_mac_change(port->phy_link, true); + else + netif_carrier_on(port->dev); + if (!delayed_work_pending(caching_dw)) queue_delayed_work(prestera_wq, caching_dw, 0); } else if (netif_running(port->dev) && netif_carrier_ok(port->dev)) { - netif_carrier_off(port->dev); + if (port->phy_link) + phylink_mac_change(port->phy_link, false); + else + netif_carrier_off(port->dev); + if (delayed_work_pending(caching_dw)) cancel_delayed_work(caching_dw); } @@ -571,19 +841,20 @@ static void prestera_event_handlers_unregister(struct prestera_switch *sw) static int prestera_switch_set_base_mac_addr(struct prestera_switch *sw) { struct device_node *base_mac_np; - struct device_node *np; int ret; - np = of_find_compatible_node(NULL, NULL, "marvell,prestera"); - base_mac_np = of_parse_phandle(np, "base-mac-provider", 0); + if (sw->np) { + base_mac_np = of_parse_phandle(sw->np, "base-mac-provider", 0); + if (base_mac_np) { + ret = of_get_mac_address(base_mac_np, sw->base_mac); + of_node_put(base_mac_np); + } + } - ret = of_get_mac_address(base_mac_np, sw->base_mac); - if (ret) { + if (!is_valid_ether_addr(sw->base_mac) || ret) { eth_random_addr(sw->base_mac); dev_info(prestera_dev(sw), "using random base mac address\n"); } - of_node_put(base_mac_np); - of_node_put(np); return prestera_hw_switch_mac_set(sw, sw->base_mac); } @@ -1083,6 +1354,8 @@ static int prestera_switch_init(struct prestera_switch *sw) { int err; + sw->np = of_find_compatible_node(NULL, NULL, "marvell,prestera"); + err = prestera_hw_switch_init(sw); if (err) { dev_err(prestera_dev(sw), "Failed to init Switch device\n"); @@ -1183,6 +1456,7 @@ static void prestera_switch_fini(struct prestera_switch *sw) prestera_router_fini(sw); prestera_netdev_event_handler_unregister(sw); prestera_hw_switch_fini(sw); + of_node_put(sw->np); } int prestera_device_register(struct prestera_device *dev) -- 2.17.1