This patch adds support for Cisco Meraki MR33 hardware highlights: SOC: IPQ4029 Quad-Core ARMv7 Processor rev 5 (v7l) Cortex-A7 DRAM: 256 MiB DDR3L-1600 @ 627 MHz Micron MT41K128M16JT-125IT NAND: 128 MiB SLC NAND Spansion S34ML01G200TFV00 (106 MiB usable) ETH: Qualcomm Atheros AR8035 Gigabit PHY (1 x LAN/WAN) + PoE WLAN1: QCA9887 (168c:0050) PCIe 1x1:1 802.11abgn ac Dualband VHT80 WLAN2: Qualcomm Atheros QCA4029 2.4GHz 802.11bgn 2:2x2 WLAN3: Qualcomm Atheros QCA4029 5GHz 802.11a/n/ac 2:2x2 VHT80 LEDS: 1 x Programmable RGB+White Status LED (driven by Ti LP5562 on i2c-1) 1 x Orange LED Fault Indicator (shared with LP5562) 2 x LAN Activity / Speed LEDs (On the RJ45 Port) BUTTON: one Reset button MISC: Bluetooth LE Ti cc2650 PG2.3 4x4mm - BL_CONFIG at 0x0001FFD8 AT24C64 8KiB EEPROM Kensington Lock Serial: WARNING: The serial port needs a TTL/RS-232 3V3 level converter! The Serial setting is 115200-8-N-1. The board has a populated 1x4 0.1" header with half-height/low profile pins. The pinout is: VCC (little white arrow), RX, TX, GND. Flashing needs a serial adaptor, as well as patched ubootwrite utility (needs Little-Endian support). And a modified u-boot (enabled Ethernet). Meraki's original u-boot source can be found in: <https://github.com/riptidewave93/meraki-uboot/tree/mr33-20170427> Add images to do an installation via bootloader: 0. open up the MR33 and connect the serial console. 1. start the 2nd stage bootloader transfer from client pc: # ubootwrite.py --write=mr33-uboot.bin (The ubootwrite tool will interrupt the boot-process and hence it needs to listen for cues. If the connection is bad (due to the low-profile pins), the tool can fail multiple times and in weird ways. If you are not sure, just use a terminal program and see what the device is doing there. 2. power on the MR33 (with ethernet + serial cables attached) Warning: Make sure you do this in a private LAN that has no connection to the internet. - let it upload the u-boot this can take 250-300 seconds - 3. use a tftp client (in binary mode!) on your PC to upload the sysupgrade.bin (the u-boot is listening on 192.168.1.1) # tftp 192.168.1.1 binary put openwrt-ipq40xx-meraki_mr33-squashfs-sysupgrade.bin 4. wait for it to reboot 5. connect to your MR33 via ssh on 192.168.1.1 For more detailed instructions, please take a look at the: "Flashing Instructions for the MR33" PDF. This can be found on the wiki: <https://openwrt.org/toh/meraki/mr33> (A link to the mr33-uboot.bin + the modified ubootwrite is also there) Thanks to Jerome C. for sending an MR33 to Chris. Signed-off-by: Chris Blake <chrisrblake93@gmail.com> Signed-off-by: Mathias Kresin <dev@kresin.me> Signed-off-by: Christian Lamparter <chunkeey@gmail.com>master
parent
4f4fc993db
commit
4943afd781
Binary file not shown.
@ -0,0 +1,14 @@ |
||||
#!/bin/sh |
||||
|
||||
. /lib/functions.sh |
||||
|
||||
preinit_set_mac_address() { |
||||
case $(board_name) in |
||||
meraki,mr33) |
||||
mac_lan=$(get_mac_binary "/sys/bus/i2c/devices/0-0050/eeprom" 102) |
||||
[ -n "$mac_lan" ] && ip link set dev eth0 address "$mac_lan" |
||||
;; |
||||
esac |
||||
} |
||||
|
||||
boot_hook_add preinit_main preinit_set_mac_address |
@ -0,0 +1,403 @@ |
||||
/* |
||||
* Device Tree Source for Meraki MR33 (Stinkbug) |
||||
* |
||||
* Copyright (C) 2017 Chris Blake <chrisrblake93@gmail.com> |
||||
* Copyright (C) 2017 Christian Lamparter <chunkeey@googlemail.com> |
||||
* |
||||
* Based on Cisco Meraki DTS from GPL release r25-linux-3.14-20170427 |
||||
* |
||||
* This file is licensed under the terms of the GNU General Public |
||||
* License version 2. This program is licensed "as is" without |
||||
* any warranty of any kind, whether express or implied. |
||||
*/ |
||||
|
||||
#include "qcom-ipq4019.dtsi" |
||||
#include "qcom-ipq4019-bus.dtsi" |
||||
#include <dt-bindings/gpio/gpio.h> |
||||
#include <dt-bindings/input/input.h> |
||||
#include <dt-bindings/soc/qcom,tcsr.h> |
||||
|
||||
/ { |
||||
model = "Meraki MR33 Access Point"; |
||||
compatible = "meraki,mr33", "qcom,ipq4019"; |
||||
|
||||
aliases { |
||||
led-boot = &status_green; |
||||
led-failsafe = &status_red; |
||||
led-running = &status_green; |
||||
led-upgrade = &power_orange; |
||||
}; |
||||
|
||||
/* Do we really need this defined? */ |
||||
memory { |
||||
device_type = "memory"; |
||||
reg = <0x80000000 0x10000000>; |
||||
}; |
||||
|
||||
reserved-memory { |
||||
#address-cells = <0x1>; |
||||
#size-cells = <0x1>; |
||||
ranges; |
||||
|
||||
tz_apps@87b80000 { |
||||
reg = <0x87b80000 0x280000>; |
||||
reusable; |
||||
}; |
||||
|
||||
smem@87e00000 { |
||||
reg = <0x87e00000 0x080000>; |
||||
no-map; |
||||
}; |
||||
|
||||
tz@87e80000 { |
||||
reg = <0x87e80000 0x180000>; |
||||
no-map; |
||||
}; |
||||
}; |
||||
|
||||
soc { |
||||
mdio@90000 { |
||||
status = "okay"; |
||||
pinctrl-0 = <&mdio_pins>; |
||||
pinctrl-names = "default"; |
||||
phy-reset-gpio = <&tlmm 47 0>; |
||||
/delete-node/ ethernet-phy@0; |
||||
/delete-node/ ethernet-phy@2; |
||||
/delete-node/ ethernet-phy@3; |
||||
/delete-node/ ethernet-phy@4; |
||||
}; |
||||
|
||||
/* It is a 56-bit counter that supplies the count to the ARM arch |
||||
timers and without upstream driver */ |
||||
counter@4a1000 { |
||||
compatible = "qcom,qca-gcnt"; |
||||
reg = <0x4a1000 0x4>; |
||||
}; |
||||
|
||||
ess_tcsr@1953000 { |
||||
compatible = "qcom,tcsr"; |
||||
reg = <0x1953000 0x1000>; |
||||
qcom,ess-interface-select = <TCSR_ESS_PSGMII_RGMII5>; |
||||
}; |
||||
|
||||
tcsr@1949000 { |
||||
compatible = "qcom,tcsr"; |
||||
reg = <0x1949000 0x100>; |
||||
qcom,wifi_glb_cfg = <TCSR_WIFI_GLB_CFG>; |
||||
}; |
||||
|
||||
tcsr@1957000 { |
||||
compatible = "qcom,tcsr"; |
||||
reg = <0x1957000 0x100>; |
||||
qcom,wifi_noc_memtype_m0_m2 = <TCSR_WIFI_NOC_MEMTYPE_M0_M2>; |
||||
}; |
||||
|
||||
serial@78af000 { |
||||
pinctrl-0 = <&serial_0_pins>; |
||||
pinctrl-names = "default"; |
||||
status = "okay"; |
||||
}; |
||||
|
||||
serial@78b0000 { |
||||
pinctrl-0 = <&serial_1_pins>; |
||||
pinctrl-names = "default"; |
||||
status = "okay"; |
||||
|
||||
bluetooth { |
||||
compatible = "ti,cc2650"; |
||||
enable-gpios = <&tlmm 12 GPIO_ACTIVE_LOW>; |
||||
}; |
||||
}; |
||||
|
||||
crypto@8e3a000 { |
||||
status = "okay"; |
||||
}; |
||||
|
||||
watchdog@b017000 { |
||||
status = "okay"; |
||||
}; |
||||
|
||||
ess-switch@c000000 { |
||||
switch_mac_mode = <0x3>; /* mac mode for RGMII RMII */ |
||||
switch_lan_bmp = <0x0>; /* lan port bitmap */ |
||||
switch_wan_bmp = <0x10>; /* wan port bitmap */ |
||||
}; |
||||
|
||||
edma@c080000 { |
||||
qcom,single-phy; |
||||
qcom,num_gmac = <1>; |
||||
phy-mode = "rgmii-rxid"; |
||||
status = "okay"; |
||||
}; |
||||
}; |
||||
|
||||
gpio-keys { |
||||
compatible = "gpio-keys"; |
||||
|
||||
reset { |
||||
label = "reset"; |
||||
gpios = <&tlmm 18 GPIO_ACTIVE_LOW>; |
||||
linux,code = <KEY_RESTART>; |
||||
}; |
||||
}; |
||||
|
||||
gpio-leds { |
||||
compatible = "gpio-leds"; |
||||
|
||||
power_orange: power { |
||||
label = "mr33:orange:power"; |
||||
gpios = <&tlmm 49 GPIO_ACTIVE_LOW>; |
||||
panic-indicator; |
||||
}; |
||||
}; |
||||
}; |
||||
|
||||
&blsp_dma { |
||||
status = "okay"; |
||||
}; |
||||
|
||||
&cryptobam { |
||||
status = "okay"; |
||||
}; |
||||
|
||||
&gmac0 { |
||||
qcom,phy_mdio_addr = <1>; |
||||
qcom,poll_required = <1>; |
||||
vlan_tag = <0 0x20>; |
||||
}; |
||||
|
||||
&i2c_0 { |
||||
pinctrl-0 = <&i2c_0_pins>; |
||||
pinctrl-names = "default"; |
||||
status = "okay"; |
||||
at24@50 { |
||||
compatible = "atmel,24c64"; |
||||
pagesize = <32>; |
||||
reg = <0x50>; |
||||
read-only; /* This holds our MAC & Meraki board-data */ |
||||
}; |
||||
}; |
||||
|
||||
&i2c_1 { |
||||
pinctrl-0 = <&i2c_1_pins>; |
||||
pinctrl-names = "default"; |
||||
status = "okay"; |
||||
|
||||
lp5562@30 { |
||||
enable-gpio = <&tlmm 48 GPIO_ACTIVE_HIGH>; |
||||
compatible = "ti,lp5562"; |
||||
clock-mode = /bits/8 <2>; |
||||
reg = <0x30>; |
||||
|
||||
/* RGB led */ |
||||
status_red: chan0 { |
||||
chan-name = "mr33:red:status"; |
||||
led-cur = /bits/ 8 <0x20>; |
||||
max-cur = /bits/ 8 <0x60>; |
||||
}; |
||||
|
||||
status_green: chan1 { |
||||
chan-name = "mr33:green:status"; |
||||
led-cur = /bits/ 8 <0x20>; |
||||
max-cur = /bits/ 8 <0x60>; |
||||
}; |
||||
|
||||
chan2 { |
||||
chan-name = "mr33:blue:status"; |
||||
led-cur = /bits/ 8 <0x20>; |
||||
max-cur = /bits/ 8 <0x60>; |
||||
}; |
||||
|
||||
chan3 { |
||||
chan-name = "mr33:white:status"; |
||||
led-cur = /bits/ 8 <0x20>; |
||||
max-cur = /bits/ 8 <0x60>; |
||||
}; |
||||
}; |
||||
}; |
||||
|
||||
&nand { |
||||
pinctrl-0 = <&nand_pins>; |
||||
pinctrl-names = "default"; |
||||
status = "okay"; |
||||
|
||||
nand@0 { |
||||
partitions { |
||||
compatible = "fixed-partitions"; |
||||
#address-cells = <1>; |
||||
#size-cells = <1>; |
||||
|
||||
partition@0 { |
||||
label = "sbl1"; |
||||
reg = <0x000000000000 0x000000100000>; |
||||
read-only; |
||||
}; |
||||
partition@1 { |
||||
label = "mibib"; |
||||
reg = <0x000000100000 0x000000100000>; |
||||
read-only; |
||||
}; |
||||
partition@2 { |
||||
label = "bootconfig"; |
||||
reg = <0x000000200000 0x000000100000>; |
||||
read-only; |
||||
}; |
||||
partition@3 { |
||||
label = "qsee"; |
||||
reg = <0x000000300000 0x000000100000>; |
||||
read-only; |
||||
}; |
||||
partition@4 { |
||||
label = "qsee_alt"; |
||||
reg = <0x000000400000 0x000000100000>; |
||||
read-only; |
||||
}; |
||||
partition@5 { |
||||
label = "cdt"; |
||||
reg = <0x000000500000 0x000000080000>; |
||||
read-only; |
||||
}; |
||||
partition@6 { |
||||
label = "cdt_alt"; |
||||
reg = <0x000000580000 0x000000080000>; |
||||
read-only; |
||||
}; |
||||
partition@7 { |
||||
label = "ddrparams"; |
||||
reg = <0x000000600000 0x000000080000>; |
||||
read-only; |
||||
}; |
||||
partition@8 { |
||||
label = "u-boot"; |
||||
reg = <0x000000700000 0x000000200000>; |
||||
read-only; |
||||
}; |
||||
partition@9 { |
||||
label = "u-boot-backup"; |
||||
reg = <0x000000900000 0x000000200000>; |
||||
read-only; |
||||
}; |
||||
partition@10 { |
||||
label = "ART"; |
||||
reg = <0x000000b00000 0x000000080000>; |
||||
read-only; |
||||
}; |
||||
partition@11 { |
||||
label = "ubi"; |
||||
reg = <0x000000c00000 0x000007000000>; |
||||
}; |
||||
}; |
||||
}; |
||||
}; |
||||
|
||||
&pcie0 { |
||||
status = "okay"; |
||||
perst-gpio = <&tlmm 38 GPIO_ACTIVE_LOW>; |
||||
wake-gpio = <&tlmm 50 GPIO_ACTIVE_LOW>; |
||||
}; |
||||
|
||||
&qpic_bam { |
||||
status = "okay"; |
||||
}; |
||||
|
||||
&tlmm { |
||||
/* |
||||
* GPIO43 should be 0/1 whenever the unit is |
||||
* powered through PoE or AC-Adapter. |
||||
* That said, playing with this seems to |
||||
* reset the AP. |
||||
*/ |
||||
|
||||
mdio_pins: mdio_pinmux { |
||||
mux_1 { |
||||
pins = "gpio6"; |
||||
function = "mdio"; |
||||
bias-pull-up; |
||||
}; |
||||
mux_2 { |
||||
pins = "gpio7"; |
||||
function = "mdc"; |
||||
bias-pull-up; |
||||
}; |
||||
}; |
||||
|
||||
serial_0_pins: serial_pinmux { |
||||
mux { |
||||
pins = "gpio16", "gpio17"; |
||||
function = "blsp_uart0"; |
||||
bias-disable; |
||||
}; |
||||
}; |
||||
|
||||
serial_1_pins: serial1_pinmux { |
||||
mux { |
||||
/* We use the i2c-0 pins for serial_1 */ |
||||
pins = "gpio8", "gpio9"; |
||||
function = "blsp_uart1"; |
||||
bias-disable; |
||||
}; |
||||
}; |
||||
|
||||
i2c_0_pins: i2c_0_pinmux { |
||||
pinmux { |
||||
function = "blsp_i2c0"; |
||||
pins = "gpio20", "gpio21"; |
||||
}; |
||||
pinconf { |
||||
pins = "gpio20", "gpio21"; |
||||
drive-strength = <16>; |
||||
bias-disable; |
||||
}; |
||||
}; |
||||
|
||||
i2c_1_pins: i2c_1_pinmux { |
||||
pinmux { |
||||
function = "blsp_i2c1"; |
||||
pins = "gpio34", "gpio35"; |
||||
}; |
||||
pinconf { |
||||
pins = "gpio34", "gpio35"; |
||||
drive-strength = <16>; |
||||
bias-disable; |
||||
}; |
||||
}; |
||||
|
||||
nand_pins: nand_pins { |
||||
/* |
||||
* There are 18 pins. 15 pins are common between LCD and NAND. |
||||
* The QPIC controller arbitrates between LCD and NAND. Of the |
||||
* remaining 4, 2 are for NAND and 2 are for LCD exclusively. |
||||
* |
||||
* The meraki source hints that the bluetooth module claims |
||||
* pin 52 as well. But sadly, there's no data whenever this |
||||
* is a NAND or LCD exclusive pin or not. |
||||
*/ |
||||
|
||||
pullups { |
||||
pins = "gpio52", "gpio53", "gpio58", |
||||
"gpio59"; |
||||
function = "qpic"; |
||||
bias-pull-up; |
||||
}; |
||||
|
||||
pulldowns { |
||||
pins = "gpio54", "gpio55", "gpio56", |
||||
"gpio57", "gpio60", "gpio61", |
||||
"gpio62", "gpio63", "gpio64", |
||||
"gpio65", "gpio66", "gpio67", |
||||
"gpio68", "gpio69"; |
||||
function = "qpic"; |
||||
bias-pull-down; |
||||
}; |
||||
}; |
||||
}; |
||||
|
||||
&wifi0 { |
||||
status = "okay"; |
||||
/* qcom,ath10k-calibration-variant = "MERAKI-MR33"; */ |
||||
}; |
||||
|
||||
&wifi1 { |
||||
status = "okay"; |
||||
/* qcom,ath10k-calibration-variant = "MERAKI-MR33"; */ |
||||
}; |
@ -0,0 +1,340 @@ |
||||
--- a/drivers/net/ethernet/qualcomm/essedma/edma_axi.c
|
||||
+++ b/drivers/net/ethernet/qualcomm/essedma/edma_axi.c
|
||||
@@ -17,6 +17,11 @@
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_net.h>
|
||||
#include <linux/timer.h>
|
||||
+#include <linux/of_platform.h>
|
||||
+#include <linux/of_address.h>
|
||||
+#include <linux/clk.h>
|
||||
+#include <linux/string.h>
|
||||
+#include <linux/reset.h>
|
||||
#include "edma.h"
|
||||
#include "ess_edma.h"
|
||||
|
||||
@@ -83,7 +88,103 @@ void edma_read_reg(u16 reg_addr, volatil
|
||||
*reg_value = readl((void __iomem *)(edma_hw_addr + reg_addr));
|
||||
}
|
||||
|
||||
-/* edma_change_tx_coalesce()
|
||||
+static void ess_write_reg(struct edma_common_info *edma, u16 reg_addr, u32 reg_value)
|
||||
+{
|
||||
+ writel(reg_value, ((void __iomem *)
|
||||
+ ((unsigned long)edma->ess_hw_addr + reg_addr)));
|
||||
+}
|
||||
+
|
||||
+static void ess_read_reg(struct edma_common_info *edma, u16 reg_addr,
|
||||
+ volatile u32 *reg_value)
|
||||
+{
|
||||
+ *reg_value = readl((void __iomem *)
|
||||
+ ((unsigned long)edma->ess_hw_addr + reg_addr));
|
||||
+}
|
||||
+
|
||||
+static int ess_reset(struct edma_common_info *edma)
|
||||
+{
|
||||
+ struct device_node *switch_node = NULL;
|
||||
+ struct reset_control *ess_rst;
|
||||
+ u32 regval;
|
||||
+
|
||||
+ switch_node = of_find_node_by_name(NULL, "ess-switch");
|
||||
+ if (!switch_node) {
|
||||
+ pr_err("switch-node not found\n");
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ ess_rst = of_reset_control_get(switch_node, "ess_rst");
|
||||
+ of_node_put(switch_node);
|
||||
+
|
||||
+ if (IS_ERR(ess_rst)) {
|
||||
+ pr_err("failed to find ess_rst!\n");
|
||||
+ return -ENOENT;
|
||||
+ }
|
||||
+
|
||||
+ reset_control_assert(ess_rst);
|
||||
+ msleep(10);
|
||||
+ reset_control_deassert(ess_rst);
|
||||
+ msleep(100);
|
||||
+ reset_control_put(ess_rst);
|
||||
+
|
||||
+ /* Enable only port 5 <--> port 0
|
||||
+ * bits 0:6 bitmap of ports it can fwd to */
|
||||
+#define SET_PORT_BMP(r,v) \
|
||||
+ ess_read_reg(edma, r, ®val); \
|
||||
+ ess_write_reg(edma, r, ((regval & ~0x3F) | v));
|
||||
+
|
||||
+ SET_PORT_BMP(ESS_PORT0_LOOKUP_CTRL,0x20);
|
||||
+ SET_PORT_BMP(ESS_PORT1_LOOKUP_CTRL,0x00);
|
||||
+ SET_PORT_BMP(ESS_PORT2_LOOKUP_CTRL,0x00);
|
||||
+ SET_PORT_BMP(ESS_PORT3_LOOKUP_CTRL,0x00);
|
||||
+ SET_PORT_BMP(ESS_PORT4_LOOKUP_CTRL,0x00);
|
||||
+ SET_PORT_BMP(ESS_PORT5_LOOKUP_CTRL,0x01);
|
||||
+ ess_write_reg(edma, ESS_RGMII_CTRL, 0x400);
|
||||
+ ess_write_reg(edma, ESS_PORT0_STATUS, ESS_PORT_1G_FDX);
|
||||
+ ess_write_reg(edma, ESS_PORT5_STATUS, ESS_PORT_1G_FDX);
|
||||
+ ess_write_reg(edma, ESS_PORT0_HEADER_CTRL, 0);
|
||||
+#undef SET_PORT_BMP
|
||||
+
|
||||
+ /* forward multicast and broadcast frames to CPU */
|
||||
+ ess_write_reg(edma, ESS_FWD_CTRL1,
|
||||
+ (ESS_PORTS_ALL << ESS_FWD_CTRL1_UC_FLOOD_S) |
|
||||
+ (ESS_PORTS_ALL << ESS_FWD_CTRL1_MC_FLOOD_S) |
|
||||
+ (ESS_PORTS_ALL << ESS_FWD_CTRL1_BC_FLOOD_S));
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+void ess_set_port_status_speed(struct edma_common_info *edma,
|
||||
+ struct phy_device *phydev, uint8_t port_id)
|
||||
+{
|
||||
+ uint16_t reg_off = ESS_PORT0_STATUS + (4 * port_id);
|
||||
+ uint32_t reg_val = 0;
|
||||
+
|
||||
+ ess_read_reg(edma, reg_off, ®_val);
|
||||
+
|
||||
+ /* reset the speed bits [0:1] */
|
||||
+ reg_val &= ~ESS_PORT_STATUS_SPEED_INV;
|
||||
+
|
||||
+ /* set the new speed */
|
||||
+ switch(phydev->speed) {
|
||||
+ case SPEED_1000: reg_val |= ESS_PORT_STATUS_SPEED_1000; break;
|
||||
+ case SPEED_100: reg_val |= ESS_PORT_STATUS_SPEED_100; break;
|
||||
+ case SPEED_10: reg_val |= ESS_PORT_STATUS_SPEED_10; break;
|
||||
+ default: reg_val |= ESS_PORT_STATUS_SPEED_INV; break;
|
||||
+ }
|
||||
+
|
||||
+ /* check full/half duplex */
|
||||
+ if (phydev->duplex) {
|
||||
+ reg_val |= ESS_PORT_STATUS_DUPLEX_MODE;
|
||||
+ } else {
|
||||
+ reg_val &= ~ESS_PORT_STATUS_DUPLEX_MODE;
|
||||
+ }
|
||||
+
|
||||
+ ess_write_reg(edma, reg_off, reg_val);
|
||||
+}
|
||||
+
|
||||
+/*
|
||||
+ * edma_change_tx_coalesce()
|
||||
* change tx interrupt moderation timer
|
||||
*/
|
||||
void edma_change_tx_coalesce(int usecs)
|
||||
@@ -551,6 +652,31 @@ static struct ctl_table edma_table[] = {
|
||||
{}
|
||||
};
|
||||
|
||||
+static int ess_parse(struct edma_common_info *edma)
|
||||
+{
|
||||
+ struct device_node *switch_node;
|
||||
+ int ret = -EINVAL;
|
||||
+
|
||||
+ switch_node = of_find_node_by_name(NULL, "ess-switch");
|
||||
+ if (!switch_node) {
|
||||
+ pr_err("cannot find ess-switch node\n");
|
||||
+ goto out;
|
||||
+ }
|
||||
+
|
||||
+ edma->ess_hw_addr = of_io_request_and_map(switch_node,
|
||||
+ 0, KBUILD_MODNAME);
|
||||
+ if (!edma->ess_hw_addr) {
|
||||
+ pr_err("%s ioremap fail.", __func__);
|
||||
+ goto out;
|
||||
+ }
|
||||
+
|
||||
+ edma->ess_clk = of_clk_get_by_name(switch_node, "ess_clk");
|
||||
+ ret = clk_prepare_enable(edma->ess_clk);
|
||||
+out:
|
||||
+ of_node_put(switch_node);
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
/* edma_axi_netdev_ops
|
||||
* Describe the operations supported by registered netdevices
|
||||
*
|
||||
@@ -786,6 +912,17 @@ static int edma_axi_probe(struct platfor
|
||||
miibus = mdio_data->mii_bus;
|
||||
}
|
||||
|
||||
+ if (of_property_read_bool(np, "qcom,single-phy") &&
|
||||
+ edma_cinfo->num_gmac == 1) {
|
||||
+ err = ess_parse(edma_cinfo);
|
||||
+ if (!err)
|
||||
+ err = ess_reset(edma_cinfo);
|
||||
+ if (err)
|
||||
+ goto err_single_phy_init;
|
||||
+ else
|
||||
+ edma_cinfo->is_single_phy = true;
|
||||
+ }
|
||||
+
|
||||
for_each_available_child_of_node(np, pnp) {
|
||||
const char *mac_addr;
|
||||
|
||||
@@ -1074,11 +1211,15 @@ static int edma_axi_probe(struct platfor
|
||||
|
||||
for (i = 0; i < edma_cinfo->num_gmac; i++) {
|
||||
if (adapter[i]->poll_required) {
|
||||
+ int phy_mode = of_get_phy_mode(np);
|
||||
+
|
||||
+ if (phy_mode < 0)
|
||||
+ phy_mode = PHY_INTERFACE_MODE_SGMII;
|
||||
adapter[i]->phydev =
|
||||
phy_connect(edma_netdev[i],
|
||||
(const char *)adapter[i]->phy_id,
|
||||
&edma_adjust_link,
|
||||
- PHY_INTERFACE_MODE_SGMII);
|
||||
+ phy_mode);
|
||||
if (IS_ERR(adapter[i]->phydev)) {
|
||||
dev_dbg(&pdev->dev, "PHY attach FAIL");
|
||||
err = -EIO;
|
||||
@@ -1125,6 +1266,9 @@ err_rmap_alloc_fail:
|
||||
for (i = 0; i < edma_cinfo->num_gmac; i++)
|
||||
unregister_netdev(edma_netdev[i]);
|
||||
err_register:
|
||||
+err_single_phy_init:
|
||||
+ iounmap(edma_cinfo->ess_hw_addr);
|
||||
+ clk_disable_unprepare(edma_cinfo->ess_clk);
|
||||
err_mdiobus_init_fail:
|
||||
edma_free_rx_rings(edma_cinfo);
|
||||
err_rx_rinit:
|
||||
@@ -1185,6 +1329,8 @@ static int edma_axi_remove(struct platfo
|
||||
del_timer_sync(&edma_stats_timer);
|
||||
edma_free_irqs(adapter);
|
||||
unregister_net_sysctl_table(edma_cinfo->edma_ctl_table_hdr);
|
||||
+ iounmap(edma_cinfo->ess_hw_addr);
|
||||
+ clk_disable_unprepare(edma_cinfo->ess_clk);
|
||||
edma_free_tx_resources(edma_cinfo);
|
||||
edma_free_rx_resources(edma_cinfo);
|
||||
edma_free_tx_rings(edma_cinfo);
|
||||
--- a/drivers/net/ethernet/qualcomm/essedma/edma.c
|
||||
+++ b/drivers/net/ethernet/qualcomm/essedma/edma.c
|
||||
@@ -161,8 +161,10 @@ static void edma_configure_rx(struct edm
|
||||
/* Set Rx FIFO threshold to start to DMA data to host */
|
||||
rxq_ctrl_data = EDMA_FIFO_THRESH_128_BYTE;
|
||||
|
||||
- /* Set RX remove vlan bit */
|
||||
- rxq_ctrl_data |= EDMA_RXQ_CTRL_RMV_VLAN;
|
||||
+ if (!edma_cinfo->is_single_phy) {
|
||||
+ /* Set RX remove vlan bit */
|
||||
+ rxq_ctrl_data |= EDMA_RXQ_CTRL_RMV_VLAN;
|
||||
+ }
|
||||
|
||||
edma_write_reg(EDMA_REG_RXQ_CTRL, rxq_ctrl_data);
|
||||
}
|
||||
@@ -1295,6 +1297,10 @@ void edma_adjust_link(struct net_device
|
||||
if (status == __EDMA_LINKUP && adapter->link_state == __EDMA_LINKDOWN) {
|
||||
dev_info(&adapter->pdev->dev, "%s: GMAC Link is up with phy_speed=%d\n", netdev->name, phydev->speed);
|
||||
adapter->link_state = __EDMA_LINKUP;
|
||||
+ if (adapter->edma_cinfo->is_single_phy) {
|
||||
+ ess_set_port_status_speed(adapter->edma_cinfo, phydev,
|
||||
+ ffs(adapter->dp_bitmap) - 1);
|
||||
+ }
|
||||
netif_carrier_on(netdev);
|
||||
if (netif_running(netdev))
|
||||
netif_tx_wake_all_queues(netdev);
|
||||
@@ -1388,10 +1394,12 @@ netdev_tx_t edma_xmit(struct sk_buff *sk
|
||||
}
|
||||
|
||||
/* Check and mark VLAN tag offload */
|
||||
- if (skb_vlan_tag_present(skb))
|
||||
- flags_transmit |= EDMA_VLAN_TX_TAG_INSERT_FLAG;
|
||||
- else if (adapter->default_vlan_tag)
|
||||
- flags_transmit |= EDMA_VLAN_TX_TAG_INSERT_DEFAULT_FLAG;
|
||||
+ if (!adapter->edma_cinfo->is_single_phy) {
|
||||
+ if (unlikely(skb_vlan_tag_present(skb)))
|
||||
+ flags_transmit |= EDMA_VLAN_TX_TAG_INSERT_FLAG;
|
||||
+ else if (adapter->default_vlan_tag)
|
||||
+ flags_transmit |= EDMA_VLAN_TX_TAG_INSERT_DEFAULT_FLAG;
|
||||
+ }
|
||||
|
||||
/* Check and mark checksum offload */
|
||||
if (likely(skb->ip_summed == CHECKSUM_PARTIAL))
|
||||
--- a/drivers/net/ethernet/qualcomm/essedma/edma.h
|
||||
+++ b/drivers/net/ethernet/qualcomm/essedma/edma.h
|
||||
@@ -31,6 +31,7 @@
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_device.h>
|
||||
+#include <linux/clk.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/device.h>
|
||||
#include <linux/sysctl.h>
|
||||
@@ -331,6 +332,10 @@ struct edma_common_info {
|
||||
struct edma_hw hw; /* edma hw specific structure */
|
||||
struct edma_per_cpu_queues_info edma_percpu_info[CONFIG_NR_CPUS]; /* per cpu information */
|
||||
spinlock_t stats_lock; /* protect edma stats area for updation */
|
||||
+
|
||||
+ bool is_single_phy;
|
||||
+ void __iomem *ess_hw_addr;
|
||||
+ struct clk *ess_clk;
|
||||
};
|
||||
|
||||
/* transimit packet descriptor (tpd) ring */
|
||||
@@ -443,4 +448,6 @@ void edma_change_tx_coalesce(int usecs);
|
||||
void edma_change_rx_coalesce(int usecs);
|
||||
void edma_get_tx_rx_coalesce(u32 *reg_val);
|
||||
void edma_clear_irq_status(void);
|
||||
+void ess_set_port_status_speed(struct edma_common_info *edma_cinfo,
|
||||
+ struct phy_device *phydev, uint8_t port_id);
|
||||
#endif /* _EDMA_H_ */
|
||||
--- a/drivers/net/ethernet/qualcomm/essedma/ess_edma.h
|
||||
+++ b/drivers/net/ethernet/qualcomm/essedma/ess_edma.h
|
||||
@@ -329,4 +329,61 @@ struct edma_hw;
|
||||
#define EDMA_RRD_PRIORITY_MASK 0x7
|
||||
#define EDMA_RRD_PORT_TYPE_SHIFT 7
|
||||
#define EDMA_RRD_PORT_TYPE_MASK 0x1F
|
||||
+
|
||||
+#define ESS_RGMII_CTRL 0x0004
|
||||
+
|
||||
+/* Port status registers */
|
||||
+#define ESS_PORT0_STATUS 0x007C
|
||||
+#define ESS_PORT1_STATUS 0x0080
|
||||
+#define ESS_PORT2_STATUS 0x0084
|
||||
+#define ESS_PORT3_STATUS 0x0088
|
||||
+#define ESS_PORT4_STATUS 0x008C
|
||||
+#define ESS_PORT5_STATUS 0x0090
|
||||
+
|
||||
+#define ESS_PORT_STATUS_HDX_FLOW_CTL 0x80
|
||||
+#define ESS_PORT_STATUS_DUPLEX_MODE 0x40
|
||||
+#define ESS_PORT_STATUS_RX_FLOW_EN 0x20
|
||||
+#define ESS_PORT_STATUS_TX_FLOW_EN 0x10
|
||||
+#define ESS_PORT_STATUS_RX_MAC_EN 0x08
|
||||
+#define ESS_PORT_STATUS_TX_MAC_EN 0x04
|
||||
+#define ESS_PORT_STATUS_SPEED_INV 0x03
|
||||
+#define ESS_PORT_STATUS_SPEED_1000 0x02
|
||||
+#define ESS_PORT_STATUS_SPEED_100 0x01
|
||||
+#define ESS_PORT_STATUS_SPEED_10 0x00
|
||||
+
|
||||
+#define ESS_PORT_1G_FDX (ESS_PORT_STATUS_DUPLEX_MODE | ESS_PORT_STATUS_RX_FLOW_EN | \
|
||||
+ ESS_PORT_STATUS_TX_FLOW_EN | ESS_PORT_STATUS_RX_MAC_EN | \
|
||||
+ ESS_PORT_STATUS_TX_MAC_EN | ESS_PORT_STATUS_SPEED_1000)
|
||||
+
|
||||
+#define PHY_STATUS_REG 0x11
|
||||
+#define PHY_STATUS_SPEED 0xC000
|
||||
+#define PHY_STATUS_SPEED_SHIFT 14
|
||||
+#define PHY_STATUS_DUPLEX 0x2000
|
||||
+#define PHY_STATUS_DUPLEX_SHIFT 13
|
||||
+#define PHY_STATUS_SPEED_DUPLEX_RESOLVED 0x0800
|
||||
+#define PHY_STATUS_CARRIER 0x0400
|
||||
+#define PHY_STATUS_CARRIER_SHIFT 10
|
||||
+
|
||||
+/* Port lookup control registers */
|
||||
+#define ESS_PORT0_LOOKUP_CTRL 0x0660
|
||||
+#define ESS_PORT1_LOOKUP_CTRL 0x066C
|
||||
+#define ESS_PORT2_LOOKUP_CTRL 0x0678
|
||||
+#define ESS_PORT3_LOOKUP_CTRL 0x0684
|
||||
+#define ESS_PORT4_LOOKUP_CTRL 0x0690
|
||||
+#define ESS_PORT5_LOOKUP_CTRL 0x069C
|
||||
+
|
||||
+#define ESS_PORT0_HEADER_CTRL 0x009C
|
||||
+
|
||||
+#define ESS_PORTS_ALL 0x3f
|
||||
+
|
||||
+#define ESS_FWD_CTRL1 0x0624
|
||||
+#define ESS_FWD_CTRL1_UC_FLOOD BITS(0, 7)
|
||||
+#define ESS_FWD_CTRL1_UC_FLOOD_S 0
|
||||
+#define ESS_FWD_CTRL1_MC_FLOOD BITS(8, 7)
|
||||
+#define ESS_FWD_CTRL1_MC_FLOOD_S 8
|
||||
+#define ESS_FWD_CTRL1_BC_FLOOD BITS(16, 7)
|
||||
+#define ESS_FWD_CTRL1_BC_FLOOD_S 16
|
||||
+#define ESS_FWD_CTRL1_IGMP BITS(24, 7)
|
||||
+#define ESS_FWD_CTRL1_IGMP_S 24
|
||||
+
|
||||
#endif /* _ESS_EDMA_H_ */
|
Loading…
Reference in new issue