// SPDX-License-Identifier: GPL-2.0-or-later /* * udbg debug output routine via GELIC UDP broadcasts * * Copyright (C) 2007 Sony Computer Entertainment Inc. * Copyright 2006, 2007 Sony Corporation * Copyright (C) 2010 Hector Martin <hector@marcansoft.com> * Copyright (C) 2011 Andre Heider <a.heider@gmail.com> */ #include <linux/if_ether.h> #include <linux/etherdevice.h> #include <linux/if_vlan.h> #include <linux/ip.h> #include <linux/udp.h> #include <asm/io.h> #include <asm/udbg.h> #include <asm/lv1call.h> #define GELIC_BUS_ID 1 #define GELIC_DEVICE_ID 0 #define GELIC_DEBUG_PORT 18194 #define GELIC_MAX_MESSAGE_SIZE 1000 #define GELIC_LV1_GET_MAC_ADDRESS 1 #define GELIC_LV1_GET_VLAN_ID 4 #define GELIC_LV1_VLAN_TX_ETHERNET_0 2 #define GELIC_DESCR_DMA_STAT_MASK 0xf0000000 #define GELIC_DESCR_DMA_CARDOWNED 0xa0000000 #define GELIC_DESCR_TX_DMA_IKE 0x00080000 #define GELIC_DESCR_TX_DMA_NO_CHKSUM 0x00000000 #define GELIC_DESCR_TX_DMA_FRAME_TAIL 0x00040000 #define GELIC_DESCR_DMA_CMD_NO_CHKSUM (GELIC_DESCR_DMA_CARDOWNED | \ GELIC_DESCR_TX_DMA_IKE | \ GELIC_DESCR_TX_DMA_NO_CHKSUM) static u64 bus_addr; struct gelic_descr { /* as defined by the hardware */ __be32 buf_addr; __be32 buf_size; __be32 next_descr_addr; __be32 dmac_cmd_status; __be32 result_size; __be32 valid_size; /* all zeroes for tx */ __be32 data_status; __be32 data_error; /* all zeroes for tx */ } __attribute__((aligned(32))); struct debug_block { struct gelic_descr descr; u8 pkt[1520]; } __packed; static __iomem struct ethhdr *h_eth; static __iomem struct vlan_hdr *h_vlan; static __iomem struct iphdr *h_ip; static __iomem struct udphdr *h_udp; static __iomem char *pmsg; static __iomem char *pmsgc; static __iomem struct debug_block dbg __attribute__((aligned(32))); static int header_size; static void map_dma_mem(int bus_id, int dev_id, void *start, size_t len, u64 *real_bus_addr) { s64 result; u64 real_addr = ((u64)start) & 0x0fffffffffffffffUL; u64 real_end = real_addr + len; u64 map_start = real_addr & ~0xfff; u64 map_end = (real_end + 0xfff) & ~0xfff; u64 bus_addr = 0; u64 flags = 0xf800000000000000UL; result = lv1_allocate_device_dma_region(bus_id, dev_id, map_end - map_start, 12, 0, &bus_addr); if (result) lv1_panic(0); result = lv1_map_device_dma_region(bus_id, dev_id, map_start, bus_addr, map_end - map_start, flags); if (result) lv1_panic(0); *real_bus_addr = bus_addr + real_addr - map_start; } static int unmap_dma_mem(int bus_id, int dev_id, u64 bus_addr, size_t len) { s64 result; u64 real_bus_addr; real_bus_addr = bus_addr & ~0xfff; len += bus_addr - real_bus_addr; len = (len + 0xfff) & ~0xfff; result = lv1_unmap_device_dma_region(bus_id, dev_id, real_bus_addr, len); if (result) return result; return lv1_free_device_dma_region(bus_id, dev_id, real_bus_addr); } static void __init gelic_debug_init(void) { s64 result; u64 v2; u64 mac; u64 vlan_id; result = lv1_open_device(GELIC_BUS_ID, GELIC_DEVICE_ID, 0); if (result) lv1_panic(0); map_dma_mem(GELIC_BUS_ID, GELIC_DEVICE_ID, &dbg, sizeof(dbg), &bus_addr); memset(&dbg, 0, sizeof(dbg)); dbg.descr.buf_addr = bus_addr + offsetof(struct debug_block, pkt); wmb(); result = lv1_net_control(GELIC_BUS_ID, GELIC_DEVICE_ID, GELIC_LV1_GET_MAC_ADDRESS, 0, 0, 0, &mac, &v2); if (result) lv1_panic(0); mac <<= 16; h_eth = (struct ethhdr *)dbg.pkt; eth_broadcast_addr(h_eth->h_dest); memcpy(&h_eth->h_source, &mac, ETH_ALEN); header_size = sizeof(struct ethhdr); result = lv1_net_control(GELIC_BUS_ID, GELIC_DEVICE_ID, GELIC_LV1_GET_VLAN_ID, GELIC_LV1_VLAN_TX_ETHERNET_0, 0, 0, &vlan_id, &v2); if (!result) { h_eth->h_proto= ETH_P_8021Q; header_size += sizeof(struct vlan_hdr); h_vlan = (struct vlan_hdr *)(h_eth + 1); h_vlan->h_vlan_TCI = vlan_id; h_vlan->h_vlan_encapsulated_proto = ETH_P_IP; h_ip = (struct iphdr *)(h_vlan + 1); } else { h_eth->h_proto= 0x0800; h_ip = (struct iphdr *)(h_eth + 1); } header_size += sizeof(struct iphdr); h_ip->version = 4; h_ip->ihl = 5; h_ip->ttl = 10; h_ip->protocol = 0x11; h_ip->saddr = 0x00000000; h_ip->daddr = 0xffffffff; header_size += sizeof(struct udphdr); h_udp = (struct udphdr *)(h_ip + 1); h_udp->source = GELIC_DEBUG_PORT; h_udp->dest = GELIC_DEBUG_PORT; pmsgc = pmsg = (char *)(h_udp + 1); } static void gelic_debug_shutdown(void) { if (bus_addr) unmap_dma_mem(GELIC_BUS_ID, GELIC_DEVICE_ID, bus_addr, sizeof(dbg)); lv1_close_device(GELIC_BUS_ID, GELIC_DEVICE_ID); } static void gelic_sendbuf(int msgsize) { u16 *p; u32 sum; int i; dbg.descr.buf_size = header_size + msgsize; h_ip->tot_len = msgsize + sizeof(struct udphdr) + sizeof(struct iphdr); h_udp->len = msgsize + sizeof(struct udphdr); h_ip->check = 0; sum = 0; p = (u16 *)h_ip; for (i = 0; i < 5; i++) sum += *p++; h_ip->check = ~(sum + (sum >> 16)); dbg.descr.dmac_cmd_status = GELIC_DESCR_DMA_CMD_NO_CHKSUM | GELIC_DESCR_TX_DMA_FRAME_TAIL; dbg.descr.result_size = 0; dbg.descr.data_status = 0; wmb(); lv1_net_start_tx_dma(GELIC_BUS_ID, GELIC_DEVICE_ID, bus_addr, 0); while ((dbg.descr.dmac_cmd_status & GELIC_DESCR_DMA_STAT_MASK) == GELIC_DESCR_DMA_CARDOWNED) cpu_relax(); } static void ps3gelic_udbg_putc(char ch) { *pmsgc++ = ch; if (ch == '\n' || (pmsgc-pmsg) >= GELIC_MAX_MESSAGE_SIZE) { gelic_sendbuf(pmsgc-pmsg); pmsgc = pmsg; } } void __init udbg_init_ps3gelic(void) { gelic_debug_init(); udbg_putc = ps3gelic_udbg_putc; } void udbg_shutdown_ps3gelic(void) { udbg_putc = NULL; gelic_debug_shutdown(); } EXPORT_SYMBOL(udbg_shutdown_ps3gelic);