162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci** hppb.c: 462306a36Sopenharmony_ci** HP-PB bus driver for the NOVA and K-Class systems. 562306a36Sopenharmony_ci** 662306a36Sopenharmony_ci** (c) Copyright 2002 Ryan Bradetich 762306a36Sopenharmony_ci** (c) Copyright 2002 Hewlett-Packard Company 862306a36Sopenharmony_ci** 962306a36Sopenharmony_ci** 1062306a36Sopenharmony_ci*/ 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci#include <linux/types.h> 1362306a36Sopenharmony_ci#include <linux/init.h> 1462306a36Sopenharmony_ci#include <linux/mm.h> 1562306a36Sopenharmony_ci#include <linux/slab.h> 1662306a36Sopenharmony_ci#include <linux/dma-mapping.h> 1762306a36Sopenharmony_ci#include <linux/ioport.h> 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci#include <asm/io.h> 2062306a36Sopenharmony_ci#include <asm/hardware.h> 2162306a36Sopenharmony_ci#include <asm/parisc-device.h> 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci#include "iommu.h" 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_cistruct hppb_card { 2662306a36Sopenharmony_ci unsigned long hpa; 2762306a36Sopenharmony_ci struct resource mmio_region; 2862306a36Sopenharmony_ci struct hppb_card *next; 2962306a36Sopenharmony_ci}; 3062306a36Sopenharmony_ci 3162306a36Sopenharmony_cistatic struct hppb_card hppb_card_head = { 3262306a36Sopenharmony_ci .hpa = 0, 3362306a36Sopenharmony_ci .next = NULL, 3462306a36Sopenharmony_ci}; 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci#define IO_IO_LOW offsetof(struct bc_module, io_io_low) 3762306a36Sopenharmony_ci#define IO_IO_HIGH offsetof(struct bc_module, io_io_high) 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_ci/** 4062306a36Sopenharmony_ci * hppb_probe - Determine if the hppb driver should claim this device. 4162306a36Sopenharmony_ci * @dev: The device which has been found 4262306a36Sopenharmony_ci * 4362306a36Sopenharmony_ci * Determine if hppb driver should claim this chip (return 0) or not 4462306a36Sopenharmony_ci * (return 1). If so, initialize the chip and tell other partners in crime 4562306a36Sopenharmony_ci * they have work to do. 4662306a36Sopenharmony_ci */ 4762306a36Sopenharmony_cistatic int __init hppb_probe(struct parisc_device *dev) 4862306a36Sopenharmony_ci{ 4962306a36Sopenharmony_ci int status; 5062306a36Sopenharmony_ci struct hppb_card *card = &hppb_card_head; 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_ci while(card->next) { 5362306a36Sopenharmony_ci card = card->next; 5462306a36Sopenharmony_ci } 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci if(card->hpa) { 5762306a36Sopenharmony_ci card->next = kzalloc(sizeof(struct hppb_card), GFP_KERNEL); 5862306a36Sopenharmony_ci if(!card->next) { 5962306a36Sopenharmony_ci printk(KERN_ERR "HP-PB: Unable to allocate memory.\n"); 6062306a36Sopenharmony_ci return 1; 6162306a36Sopenharmony_ci } 6262306a36Sopenharmony_ci card = card->next; 6362306a36Sopenharmony_ci } 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci card->hpa = dev->hpa.start; 6662306a36Sopenharmony_ci card->mmio_region.name = "HP-PB Bus"; 6762306a36Sopenharmony_ci card->mmio_region.flags = IORESOURCE_MEM; 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci card->mmio_region.start = gsc_readl(dev->hpa.start + IO_IO_LOW); 7062306a36Sopenharmony_ci card->mmio_region.end = gsc_readl(dev->hpa.start + IO_IO_HIGH) - 1; 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci status = ccio_request_resource(dev, &card->mmio_region); 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_ci pr_info("Found GeckoBoa at %pap, bus space %pR,%s claimed.\n", 7562306a36Sopenharmony_ci &dev->hpa.start, 7662306a36Sopenharmony_ci &card->mmio_region, 7762306a36Sopenharmony_ci (status < 0) ? " not":"" ); 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci return 0; 8062306a36Sopenharmony_ci} 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_cistatic const struct parisc_device_id hppb_tbl[] __initconst = { 8362306a36Sopenharmony_ci { HPHW_BCPORT, HVERSION_REV_ANY_ID, 0x500, 0xc }, /* E25 and K */ 8462306a36Sopenharmony_ci { HPHW_BCPORT, 0x0, 0x501, 0xc }, /* E35 */ 8562306a36Sopenharmony_ci { HPHW_BCPORT, 0x0, 0x502, 0xc }, /* E45 */ 8662306a36Sopenharmony_ci { HPHW_BCPORT, 0x0, 0x503, 0xc }, /* E55 */ 8762306a36Sopenharmony_ci { 0, } 8862306a36Sopenharmony_ci}; 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_cistatic struct parisc_driver hppb_driver __refdata = { 9162306a36Sopenharmony_ci .name = "gecko_boa", 9262306a36Sopenharmony_ci .id_table = hppb_tbl, 9362306a36Sopenharmony_ci .probe = hppb_probe, 9462306a36Sopenharmony_ci}; 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_ci/** 9762306a36Sopenharmony_ci * hppb_init - HP-PB bus initialization procedure. 9862306a36Sopenharmony_ci * 9962306a36Sopenharmony_ci * Register this driver. 10062306a36Sopenharmony_ci */ 10162306a36Sopenharmony_cistatic int __init hppb_init(void) 10262306a36Sopenharmony_ci{ 10362306a36Sopenharmony_ci return register_parisc_driver(&hppb_driver); 10462306a36Sopenharmony_ci} 10562306a36Sopenharmony_ciarch_initcall(hppb_init); 106