| // 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); |