31e40c8ce1
MIPS32 4Kc CPU board, with support for the MIPS64 5Kc and the QED RM5261 CPU boards to follow. The cs4281 audio hasn't been tested, there are some interrupt problems with onboard the pciide, but all other on-board peripherals work. The evbmips port will support more MIPS evaluation boards in the future.
932 lines
25 KiB
C
932 lines
25 KiB
C
/* $NetBSD: pci_alignstride_bus_io_chipdep.c,v 1.1 2002/03/07 14:44:06 simonb Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 1998, 2000, 2001 The NetBSD Foundation, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
* by Jason R. Thorpe of the Numerical Aerospace Simulation Facility,
|
|
* NASA Ames Research Center.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by the NetBSD
|
|
* Foundation, Inc. and its contributors.
|
|
* 4. Neither the name of The NetBSD Foundation nor the names of its
|
|
* contributors may be used to endorse or promote products derived
|
|
* from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/*
|
|
* Copyright (c) 1995, 1996 Carnegie-Mellon University.
|
|
* All rights reserved.
|
|
*
|
|
* Author: Chris G. Demetriou
|
|
*
|
|
* Permission to use, copy, modify and distribute this software and
|
|
* its documentation is hereby granted, provided that both the copyright
|
|
* notice and this permission notice appear in all copies of the
|
|
* software, derivative works or modified versions, and any portions
|
|
* thereof, and that both notices appear in supporting documentation.
|
|
*
|
|
* CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
|
|
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND
|
|
* FOR ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
|
|
*
|
|
* Carnegie Mellon requests users of this software to return to
|
|
*
|
|
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
|
|
* School of Computer Science
|
|
* Carnegie Mellon University
|
|
* Pittsburgh PA 15213-3890
|
|
*
|
|
* any improvements or extensions that they make and grant Carnegie the
|
|
* rights to redistribute these changes.
|
|
*/
|
|
|
|
/*
|
|
* Common PCI Chipset "bus I/O" functions, for chipsets which have to
|
|
* deal with only a single PCI interface chip in a machine.
|
|
*
|
|
* uses:
|
|
* CHIP name of the 'chip' it's being compiled for.
|
|
* CHIP_IO_BASE Sparse I/O space base to use.
|
|
* CHIP_IO_EX_STORE
|
|
* If defined, device-provided static storage area
|
|
* for the I/O space extent. If this is defined,
|
|
* CHIP_IO_EX_STORE_SIZE must also be defined. If
|
|
* this is not defined, a static area will be
|
|
* declared.
|
|
* CHIP_IO_EX_STORE_SIZE
|
|
* Size of the device-provided static storage area
|
|
* for the I/O memory space extent.
|
|
*/
|
|
|
|
#include <sys/extent.h>
|
|
|
|
#define __C(A,B) __CONCAT(A,B)
|
|
#define __S(S) __STRING(S)
|
|
|
|
/* mapping/unmapping */
|
|
int __C(CHIP,_io_map) __P((void *, bus_addr_t, bus_size_t, int,
|
|
bus_space_handle_t *, int));
|
|
void __C(CHIP,_io_unmap) __P((void *, bus_space_handle_t,
|
|
bus_size_t, int));
|
|
int __C(CHIP,_io_subregion) __P((void *, bus_space_handle_t,
|
|
bus_size_t, bus_size_t, bus_space_handle_t *));
|
|
|
|
int __C(CHIP,_io_translate) __P((void *, bus_addr_t, bus_size_t,
|
|
int, struct mips_bus_space_translation *));
|
|
int __C(CHIP,_io_get_window) __P((void *, int,
|
|
struct mips_bus_space_translation *));
|
|
|
|
/* allocation/deallocation */
|
|
int __C(CHIP,_io_alloc) __P((void *, bus_addr_t, bus_addr_t,
|
|
bus_size_t, bus_size_t, bus_addr_t, int, bus_addr_t *,
|
|
bus_space_handle_t *));
|
|
void __C(CHIP,_io_free) __P((void *, bus_space_handle_t,
|
|
bus_size_t));
|
|
|
|
/* get kernel virtual address */
|
|
void * __C(CHIP,_io_vaddr) __P((void *, bus_space_handle_t));
|
|
|
|
/* mmap for user */
|
|
paddr_t __C(CHIP,_io_mmap) __P((void *, bus_addr_t, off_t, int, int));
|
|
|
|
/* barrier */
|
|
inline void __C(CHIP,_io_barrier) __P((void *, bus_space_handle_t,
|
|
bus_size_t, bus_size_t, int));
|
|
|
|
/* read (single) */
|
|
inline uint8_t __C(CHIP,_io_read_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t));
|
|
inline uint16_t __C(CHIP,_io_read_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t));
|
|
inline uint32_t __C(CHIP,_io_read_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t));
|
|
inline uint64_t __C(CHIP,_io_read_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t));
|
|
|
|
/* read multiple */
|
|
void __C(CHIP,_io_read_multi_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint8_t *, bus_size_t));
|
|
void __C(CHIP,_io_read_multi_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint16_t *, bus_size_t));
|
|
void __C(CHIP,_io_read_multi_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint32_t *, bus_size_t));
|
|
void __C(CHIP,_io_read_multi_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint64_t *, bus_size_t));
|
|
|
|
/* read region */
|
|
void __C(CHIP,_io_read_region_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint8_t *, bus_size_t));
|
|
void __C(CHIP,_io_read_region_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint16_t *, bus_size_t));
|
|
void __C(CHIP,_io_read_region_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint32_t *, bus_size_t));
|
|
void __C(CHIP,_io_read_region_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint64_t *, bus_size_t));
|
|
|
|
/* write (single) */
|
|
inline void __C(CHIP,_io_write_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint8_t));
|
|
inline void __C(CHIP,_io_write_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint16_t));
|
|
inline void __C(CHIP,_io_write_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint32_t));
|
|
inline void __C(CHIP,_io_write_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint64_t));
|
|
|
|
/* write multiple */
|
|
void __C(CHIP,_io_write_multi_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t, const uint8_t *, bus_size_t));
|
|
void __C(CHIP,_io_write_multi_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t, const uint16_t *, bus_size_t));
|
|
void __C(CHIP,_io_write_multi_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t, const uint32_t *, bus_size_t));
|
|
void __C(CHIP,_io_write_multi_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t, const uint64_t *, bus_size_t));
|
|
|
|
/* write region */
|
|
void __C(CHIP,_io_write_region_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t, const uint8_t *, bus_size_t));
|
|
void __C(CHIP,_io_write_region_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t, const uint16_t *, bus_size_t));
|
|
void __C(CHIP,_io_write_region_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t, const uint32_t *, bus_size_t));
|
|
void __C(CHIP,_io_write_region_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t, const uint64_t *, bus_size_t));
|
|
|
|
/* set multiple */
|
|
void __C(CHIP,_io_set_multi_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint8_t, bus_size_t));
|
|
void __C(CHIP,_io_set_multi_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint16_t, bus_size_t));
|
|
void __C(CHIP,_io_set_multi_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint32_t, bus_size_t));
|
|
void __C(CHIP,_io_set_multi_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint64_t, bus_size_t));
|
|
|
|
/* set region */
|
|
void __C(CHIP,_io_set_region_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint8_t, bus_size_t));
|
|
void __C(CHIP,_io_set_region_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint16_t, bus_size_t));
|
|
void __C(CHIP,_io_set_region_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint32_t, bus_size_t));
|
|
void __C(CHIP,_io_set_region_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t, uint64_t, bus_size_t));
|
|
|
|
/* copy */
|
|
void __C(CHIP,_io_copy_region_1) __P((void *, bus_space_handle_t,
|
|
bus_size_t, bus_space_handle_t, bus_size_t, bus_size_t));
|
|
void __C(CHIP,_io_copy_region_2) __P((void *, bus_space_handle_t,
|
|
bus_size_t, bus_space_handle_t, bus_size_t, bus_size_t));
|
|
void __C(CHIP,_io_copy_region_4) __P((void *, bus_space_handle_t,
|
|
bus_size_t, bus_space_handle_t, bus_size_t, bus_size_t));
|
|
void __C(CHIP,_io_copy_region_8) __P((void *, bus_space_handle_t,
|
|
bus_size_t, bus_space_handle_t, bus_size_t, bus_size_t));
|
|
|
|
#ifdef CHIP_IO_EXTENT
|
|
#ifndef CHIP_IO_EX_STORE
|
|
static long
|
|
__C(CHIP,_io_ex_storage)[EXTENT_FIXED_STORAGE_SIZE(8) / sizeof(long)];
|
|
#define CHIP_IO_EX_STORE(v) (__C(CHIP, _io_ex_storage))
|
|
#define CHIP_IO_EX_STORE_SIZE(v) (sizeof __C(CHIP, _io_ex_storage))
|
|
#endif
|
|
#endif /* CHIP_IO_EXTENT */
|
|
|
|
#ifndef CHIP_ALIGN_STRIDE
|
|
#define CHIP_ALIGN_STRIDE 0
|
|
#endif
|
|
|
|
void
|
|
__C(CHIP,_bus_io_init)(t, v)
|
|
bus_space_tag_t t;
|
|
void *v;
|
|
{
|
|
#ifdef CHIP_IO_EXTENT
|
|
struct extent *ex;
|
|
#endif
|
|
|
|
/*
|
|
* Initialize the bus space tag.
|
|
*/
|
|
|
|
/* cookie */
|
|
t->bs_cookie = v;
|
|
|
|
/* mapping/unmapping */
|
|
t->bs_map = __C(CHIP,_io_map);
|
|
t->bs_unmap = __C(CHIP,_io_unmap);
|
|
t->bs_subregion = __C(CHIP,_io_subregion);
|
|
|
|
t->bs_translate = __C(CHIP,_io_translate);
|
|
t->bs_get_window = __C(CHIP,_io_get_window);
|
|
|
|
/* allocation/deallocation */
|
|
t->bs_alloc = __C(CHIP,_io_alloc);
|
|
t->bs_free = __C(CHIP,_io_free);
|
|
|
|
/* get kernel virtual address */
|
|
t->bs_vaddr = __C(CHIP,_io_vaddr);
|
|
|
|
/* mmap for user */
|
|
t->bs_mmap = __C(CHIP,_io_mmap);
|
|
|
|
/* barrier */
|
|
t->bs_barrier = __C(CHIP,_io_barrier);
|
|
|
|
/* read (single) */
|
|
t->bs_r_1 = __C(CHIP,_io_read_1);
|
|
t->bs_r_2 = __C(CHIP,_io_read_2);
|
|
t->bs_r_4 = __C(CHIP,_io_read_4);
|
|
t->bs_r_8 = __C(CHIP,_io_read_8);
|
|
|
|
/* read multiple */
|
|
t->bs_rm_1 = __C(CHIP,_io_read_multi_1);
|
|
t->bs_rm_2 = __C(CHIP,_io_read_multi_2);
|
|
t->bs_rm_4 = __C(CHIP,_io_read_multi_4);
|
|
t->bs_rm_8 = __C(CHIP,_io_read_multi_8);
|
|
|
|
/* read region */
|
|
t->bs_rr_1 = __C(CHIP,_io_read_region_1);
|
|
t->bs_rr_2 = __C(CHIP,_io_read_region_2);
|
|
t->bs_rr_4 = __C(CHIP,_io_read_region_4);
|
|
t->bs_rr_8 = __C(CHIP,_io_read_region_8);
|
|
|
|
/* write (single) */
|
|
t->bs_w_1 = __C(CHIP,_io_write_1);
|
|
t->bs_w_2 = __C(CHIP,_io_write_2);
|
|
t->bs_w_4 = __C(CHIP,_io_write_4);
|
|
t->bs_w_8 = __C(CHIP,_io_write_8);
|
|
|
|
/* write multiple */
|
|
t->bs_wm_1 = __C(CHIP,_io_write_multi_1);
|
|
t->bs_wm_2 = __C(CHIP,_io_write_multi_2);
|
|
t->bs_wm_4 = __C(CHIP,_io_write_multi_4);
|
|
t->bs_wm_8 = __C(CHIP,_io_write_multi_8);
|
|
|
|
/* write region */
|
|
t->bs_wr_1 = __C(CHIP,_io_write_region_1);
|
|
t->bs_wr_2 = __C(CHIP,_io_write_region_2);
|
|
t->bs_wr_4 = __C(CHIP,_io_write_region_4);
|
|
t->bs_wr_8 = __C(CHIP,_io_write_region_8);
|
|
|
|
/* set multiple */
|
|
t->bs_sm_1 = __C(CHIP,_io_set_multi_1);
|
|
t->bs_sm_2 = __C(CHIP,_io_set_multi_2);
|
|
t->bs_sm_4 = __C(CHIP,_io_set_multi_4);
|
|
t->bs_sm_8 = __C(CHIP,_io_set_multi_8);
|
|
|
|
/* set region */
|
|
t->bs_sr_1 = __C(CHIP,_io_set_region_1);
|
|
t->bs_sr_2 = __C(CHIP,_io_set_region_2);
|
|
t->bs_sr_4 = __C(CHIP,_io_set_region_4);
|
|
t->bs_sr_8 = __C(CHIP,_io_set_region_8);
|
|
|
|
/* copy */
|
|
t->bs_c_1 = __C(CHIP,_io_copy_region_1);
|
|
t->bs_c_2 = __C(CHIP,_io_copy_region_2);
|
|
t->bs_c_4 = __C(CHIP,_io_copy_region_4);
|
|
t->bs_c_8 = __C(CHIP,_io_copy_region_8);
|
|
|
|
#ifdef CHIP_IO_EXTENT
|
|
/* XXX WE WANT EXTENT_NOCOALESCE, BUT WE CAN'T USE IT. XXX */
|
|
ex = extent_create(__S(__C(CHIP,_bus_io)), 0x0UL, 0xffffffffUL,
|
|
M_DEVBUF, (caddr_t)CHIP_IO_EX_STORE(v), CHIP_IO_EX_STORE_SIZE(v),
|
|
EX_NOWAIT);
|
|
extent_alloc_region(ex, 0, 0xffffffffUL, EX_NOWAIT);
|
|
|
|
#ifdef CHIP_IO_W1_BUS_START
|
|
/*
|
|
* The window may be disabled. We notice this by seeing
|
|
* -1 as the bus base address.
|
|
*/
|
|
if (CHIP_IO_W1_BUS_START(v) == (bus_addr_t) -1) {
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: this space is disabled\n");
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: freeing from 0x%lx to 0x%lx\n", CHIP_IO_W1_BUS_START(v),
|
|
(long)CHIP_IO_W1_BUS_END(v));
|
|
#endif
|
|
extent_free(ex, CHIP_IO_W1_BUS_START(v),
|
|
CHIP_IO_W1_BUS_END(v) - CHIP_IO_W1_BUS_START(v) + 1, EX_NOWAIT);
|
|
#endif
|
|
#ifdef CHIP_IO_W2_BUS_START
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: freeing from 0x%lx to 0x%lx\n", CHIP_IO_W2_BUS_START(v),
|
|
(long)CHIP_IO_W2_BUS_END(v));
|
|
#endif
|
|
extent_free(ex, CHIP_IO_W2_BUS_START(v),
|
|
CHIP_IO_W2_BUS_END(v) - CHIP_IO_W2_BUS_START(v) + 1, EX_NOWAIT);
|
|
#endif
|
|
|
|
#ifdef EXTENT_DEBUG
|
|
extent_print(ex);
|
|
#endif
|
|
CHIP_IO_EXTENT(v) = ex;
|
|
#endif /* CHIP_IO_EXTENT */
|
|
}
|
|
|
|
int
|
|
__C(CHIP,_io_translate)(v, ioaddr, iolen, flags, mbst)
|
|
void *v;
|
|
bus_addr_t ioaddr;
|
|
bus_size_t iolen;
|
|
int flags;
|
|
struct mips_bus_space_translation *mbst;
|
|
{
|
|
bus_addr_t ioend = ioaddr + (iolen - 1);
|
|
int linear = flags & BUS_SPACE_MAP_LINEAR;
|
|
|
|
/*
|
|
* Can't map i/o space linearly.
|
|
*/
|
|
if (linear)
|
|
return (EOPNOTSUPP);
|
|
|
|
#ifdef CHIP_IO_W1_BUS_START
|
|
if (ioaddr >= CHIP_IO_W1_BUS_START(v) &&
|
|
ioend <= CHIP_IO_W1_BUS_END(v))
|
|
return (__C(CHIP,_io_get_window)(v, 0, mbst));
|
|
#endif
|
|
|
|
#ifdef CHIP_IO_W2_BUS_START
|
|
if (ioaddr >= CHIP_IO_W2_BUS_START(v) &&
|
|
ioend <= CHIP_IO_W2_BUS_END(v))
|
|
return (__C(CHIP,_io_get_window)(v, 1, mbst));
|
|
#endif
|
|
|
|
#ifdef EXTENT_DEBUG
|
|
printf("\n");
|
|
#ifdef CHIP_IO_W1_BUS_START
|
|
printf("%s: window[1]=0x%lx-0x%lx\n",
|
|
__S(__C(CHIP,_io_map)), CHIP_IO_W1_BUS_START(v),
|
|
(long)CHIP_IO_W1_BUS_END(v));
|
|
#endif
|
|
#ifdef CHIP_IO_W2_BUS_START
|
|
printf("%s: window[2]=0x%lx-0x%lx\n",
|
|
__S(__C(CHIP,_io_map)), CHIP_IO_W2_BUS_START(v),
|
|
(long)CHIP_IO_W2_BUS_END(v));
|
|
#endif
|
|
#endif /* EXTENT_DEBUG */
|
|
/* No translation. */
|
|
return (EINVAL);
|
|
}
|
|
|
|
int
|
|
__C(CHIP,_io_get_window)(v, window, mbst)
|
|
void *v;
|
|
int window;
|
|
struct mips_bus_space_translation *mbst;
|
|
{
|
|
|
|
switch (window) {
|
|
#ifdef CHIP_IO_W1_BUS_START
|
|
case 0:
|
|
mbst->mbst_bus_start = CHIP_IO_W1_BUS_START(v);
|
|
mbst->mbst_bus_end = CHIP_IO_W1_BUS_END(v);
|
|
mbst->mbst_sys_start = CHIP_IO_W1_SYS_START(v);
|
|
mbst->mbst_sys_end = CHIP_IO_W1_SYS_END(v);
|
|
mbst->mbst_align_stride = CHIP_ALIGN_STRIDE;
|
|
mbst->mbst_flags = 0;
|
|
break;
|
|
#endif
|
|
|
|
#ifdef CHIP_IO_W2_BUS_START
|
|
case 1:
|
|
mbst->mbst_bus_start = CHIP_IO_W2_BUS_START(v);
|
|
mbst->mbst_bus_end = CHIP_IO_W2_BUS_END(v);
|
|
mbst->mbst_sys_start = CHIP_IO_W2_SYS_START(v);
|
|
mbst->mbst_sys_end = CHIP_IO_W2_SYS_END(v);
|
|
mbst->mbst_align_stride = CHIP_ALIGN_STRIDE;
|
|
mbst->mbst_flags = 0;
|
|
break;
|
|
#endif
|
|
|
|
default:
|
|
panic(__S(__C(CHIP,_io_get_window)) ": invalid window %d",
|
|
window);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
__C(CHIP,_io_map)(v, ioaddr, iosize, flags, iohp, acct)
|
|
void *v;
|
|
bus_addr_t ioaddr;
|
|
bus_size_t iosize;
|
|
int flags;
|
|
bus_space_handle_t *iohp;
|
|
int acct;
|
|
{
|
|
struct mips_bus_space_translation mbst;
|
|
int error;
|
|
|
|
/*
|
|
* Get the translation for this address.
|
|
*/
|
|
error = __C(CHIP,_io_translate)(v, ioaddr, iosize, flags, &mbst);
|
|
if (error)
|
|
return (error);
|
|
|
|
#ifdef CHIP_IO_EXTENT
|
|
if (acct == 0)
|
|
goto mapit;
|
|
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: allocating 0x%lx to 0x%lx\n", ioaddr, ioaddr + iosize - 1);
|
|
#endif
|
|
error = extent_alloc_region(CHIP_IO_EXTENT(v), ioaddr, iosize,
|
|
EX_NOWAIT | (CHIP_EX_MALLOC_SAFE(v) ? EX_MALLOCOK : 0));
|
|
if (error) {
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: allocation failed (%d)\n", error);
|
|
extent_print(CHIP_IO_EXTENT(v));
|
|
#endif
|
|
return (error);
|
|
}
|
|
|
|
mapit:
|
|
#endif /* CHIP_IO_EXTENT */
|
|
if (flags & BUS_SPACE_MAP_CACHEABLE)
|
|
*iohp = MIPS_PHYS_TO_KSEG0(mbst.mbst_sys_start +
|
|
(ioaddr - mbst.mbst_bus_start));
|
|
else
|
|
*iohp = MIPS_PHYS_TO_KSEG1(mbst.mbst_sys_start +
|
|
(ioaddr - mbst.mbst_bus_start));
|
|
|
|
return (0);
|
|
}
|
|
|
|
void
|
|
__C(CHIP,_io_unmap)(v, ioh, iosize, acct)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t iosize;
|
|
int acct;
|
|
{
|
|
#ifdef CHIP_IO_EXTENT
|
|
bus_addr_t ioaddr;
|
|
int error;
|
|
|
|
if (acct == 0)
|
|
return;
|
|
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: freeing handle 0x%lx for 0x%lx\n", ioh, iosize);
|
|
#endif
|
|
|
|
if (ioh >= MIPS_KSEG0_START && ioh < MIPS_KSEG1_START)
|
|
ioh = MIPS_KSEG0_TO_PHYS(ioh);
|
|
else
|
|
ioh = MIPS_KSEG1_TO_PHYS(ioh);
|
|
|
|
#ifdef CHIP_IO_W1_BUS_START
|
|
if (ioh >= CHIP_IO_W1_SYS_START(v) &&
|
|
ioh <= CHIP_IO_W1_SYS_END(v)) {
|
|
ioaddr = CHIP_IO_W1_BUS_START(v) +
|
|
(ioh - CHIP_IO_W1_SYS_START(v));
|
|
} else
|
|
#endif
|
|
#ifdef CHIP_IO_W2_BUS_START
|
|
if (ioh >= CHIP_IO_W2_SYS_START(v) &&
|
|
ioh <= CHIP_IO_W2_SYS_END(v)) {
|
|
ioaddr = CHIP_IO_W2_BUS_START(v) +
|
|
(ioh - CHIP_IO_W2_SYS_START(v));
|
|
} else
|
|
#endif
|
|
{
|
|
printf("\n");
|
|
#ifdef CHIP_IO_W1_BUS_START
|
|
printf("%s: sys window[1]=0x%lx-0x%lx\n",
|
|
__S(__C(CHIP,_io_map)), CHIP_IO_W1_SYS_START(v),
|
|
CHIP_IO_W1_SYS_END(v));
|
|
#endif
|
|
#ifdef CHIP_IO_W2_BUS_START
|
|
printf("%s: sys window[2]=0x%lx-0x%lx\n",
|
|
__S(__C(CHIP,_io_map)), CHIP_IO_W2_SYS_START(v),
|
|
CHIP_IO_W2_SYS_END(v));
|
|
#endif
|
|
panic("%s: don't know how to unmap %lx",
|
|
__S(__C(CHIP,_io_unmap)), ioh);
|
|
}
|
|
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: freeing 0x%lx to 0x%lx\n", ioaddr, ioaddr + iosize - 1);
|
|
#endif
|
|
error = extent_free(CHIP_IO_EXTENT(v), ioaddr, iosize,
|
|
EX_NOWAIT | (CHIP_EX_MALLOC_SAFE(v) ? EX_MALLOCOK : 0));
|
|
if (error) {
|
|
printf("%s: WARNING: could not unmap 0x%lx-0x%lx (error %d)\n",
|
|
__S(__C(CHIP,_io_unmap)), ioaddr, ioaddr + iosize - 1,
|
|
error);
|
|
#ifdef EXTENT_DEBUG
|
|
extent_print(CHIP_IO_EXTENT(v));
|
|
#endif
|
|
}
|
|
#endif /* CHIP_IO_EXTENT */
|
|
}
|
|
|
|
int
|
|
__C(CHIP,_io_subregion)(v, ioh, offset, size, nioh)
|
|
void *v;
|
|
bus_space_handle_t ioh, *nioh;
|
|
bus_size_t offset, size;
|
|
{
|
|
|
|
*nioh = ioh + (offset << CHIP_ALIGN_STRIDE);
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
__C(CHIP,_io_alloc)(v, rstart, rend, size, align, boundary, flags,
|
|
addrp, bshp)
|
|
void *v;
|
|
bus_addr_t rstart, rend, *addrp;
|
|
bus_size_t size, align, boundary;
|
|
int flags;
|
|
bus_space_handle_t *bshp;
|
|
{
|
|
#ifdef CHIP_IO_EXTENT
|
|
struct mips_bus_space_translation mbst;
|
|
int linear = flags & BUS_SPACE_MAP_LINEAR;
|
|
bus_addr_t ioaddr;
|
|
int error;
|
|
|
|
/*
|
|
* Can't map i/o space linearly.
|
|
*/
|
|
if (linear)
|
|
return (EOPNOTSUPP);
|
|
|
|
/*
|
|
* Do the requested allocation.
|
|
*/
|
|
#ifdef EXTENT_DEBUG
|
|
printf("\nio_alloc:\n");
|
|
printf("io: allocating from 0x%lx to 0x%lx\n", rstart, rend);
|
|
#endif
|
|
error = extent_alloc_subregion(CHIP_IO_EXTENT(v), rstart, rend,
|
|
size, align, boundary,
|
|
EX_FAST | EX_NOWAIT | (CHIP_EX_MALLOC_SAFE(v) ? EX_MALLOCOK : 0),
|
|
&ioaddr);
|
|
if (error) {
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: allocation failed (%d)\n", error);
|
|
extent_print(CHIP_IO_EXTENT(v));
|
|
#endif
|
|
return (error);
|
|
}
|
|
|
|
#ifdef EXTENT_DEBUG
|
|
printf("io: allocated 0x%lx to 0x%lx\n", ioaddr, ioaddr + size - 1);
|
|
#endif
|
|
|
|
error = __C(CHIP,_io_translate)(v, ioaddr, size, flags, &mbst);
|
|
if (error) {
|
|
(void) extent_free(CHIP_IO_EXTENT(v), ioaddr, size,
|
|
EX_NOWAIT | (CHIP_EX_MALLOC_SAFE(v) ? EX_MALLOCOK : 0));
|
|
return (error);
|
|
}
|
|
|
|
*addrp = ioaddr;
|
|
if (flags & BUS_SPACE_MAP_CACHEABLE)
|
|
*bshp = MIPS_PHYS_TO_KSEG0(mbst.mbst_sys_start +
|
|
(ioaddr - mbst.mbst_bus_start));
|
|
else
|
|
*bshp = MIPS_PHYS_TO_KSEG1(mbst.mbst_sys_start +
|
|
(ioaddr - mbst.mbst_bus_start));
|
|
|
|
return (0);
|
|
#else /* ! CHIP_IO_EXTENT */
|
|
return (EOPNOTSUPP);
|
|
#endif /* CHIP_IO_EXTENT */
|
|
}
|
|
|
|
void
|
|
__C(CHIP,_io_free)(v, bsh, size)
|
|
void *v;
|
|
bus_space_handle_t bsh;
|
|
bus_size_t size;
|
|
{
|
|
|
|
/* Unmap does all we need to do. */
|
|
__C(CHIP,_io_unmap)(v, bsh, size, 1);
|
|
}
|
|
|
|
void *
|
|
__C(CHIP,_io_vaddr)(v, bsh)
|
|
void *v;
|
|
bus_space_handle_t bsh;
|
|
{
|
|
/*
|
|
* _io_translate() catches BUS_SPACE_MAP_LINEAR,
|
|
* so we shouldn't get here
|
|
*/
|
|
panic("_io_vaddr");
|
|
}
|
|
|
|
paddr_t
|
|
__C(CHIP,_io_mmap)(v, addr, off, prot, flags)
|
|
void *v;
|
|
bus_addr_t addr;
|
|
off_t off;
|
|
int prot;
|
|
int flags;
|
|
{
|
|
|
|
/* Not supported for I/O space. */
|
|
return (-1);
|
|
}
|
|
|
|
inline void
|
|
__C(CHIP,_io_barrier)(v, h, o, l, f)
|
|
void *v;
|
|
bus_space_handle_t h;
|
|
bus_size_t o, l;
|
|
int f;
|
|
{
|
|
|
|
/* XXX XXX XXX */
|
|
if ((f & BUS_SPACE_BARRIER_WRITE) != 0)
|
|
wbflush();
|
|
}
|
|
|
|
inline uint8_t
|
|
__C(CHIP,_io_read_1)(v, ioh, off)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t off;
|
|
{
|
|
uint8_t *ptr = (void *)(ioh + (off << CHIP_ALIGN_STRIDE));
|
|
|
|
return (*ptr);
|
|
}
|
|
|
|
inline uint16_t
|
|
__C(CHIP,_io_read_2)(v, ioh, off)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t off;
|
|
{
|
|
#if CHIP_ALIGN_STRIDE >= 1
|
|
uint16_t *ptr = (void *)(ioh + (off << (CHIP_ALIGN_STRIDE - 1)));
|
|
#else
|
|
uint16_t *ptr = (void *)(ioh + off);
|
|
#endif
|
|
|
|
return (*ptr);
|
|
}
|
|
|
|
inline uint32_t
|
|
__C(CHIP,_io_read_4)(v, ioh, off)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t off;
|
|
{
|
|
#if CHIP_ALIGN_STRIDE >= 2
|
|
uint32_t *ptr = (void *)(ioh + (off << (CHIP_ALIGN_STRIDE - 2)));
|
|
#else
|
|
uint32_t *ptr = (void *)(ioh + off);
|
|
#endif
|
|
|
|
return (*ptr);
|
|
}
|
|
|
|
inline uint64_t
|
|
__C(CHIP,_io_read_8)(v, ioh, off)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t off;
|
|
{
|
|
|
|
/* XXX XXX XXX */
|
|
panic("%s not implemented", __S(__C(CHIP,_io_read_8)));
|
|
}
|
|
|
|
#define CHIP_io_read_multi_N(BYTES,TYPE) \
|
|
void \
|
|
__C(__C(CHIP,_io_read_multi_),BYTES)(v, h, o, a, c) \
|
|
void *v; \
|
|
bus_space_handle_t h; \
|
|
bus_size_t o, c; \
|
|
TYPE *a; \
|
|
{ \
|
|
\
|
|
while (c-- > 0) { \
|
|
__C(CHIP,_io_barrier)(v, h, o, sizeof *a, \
|
|
BUS_SPACE_BARRIER_READ); \
|
|
*a++ = __C(__C(CHIP,_io_read_),BYTES)(v, h, o); \
|
|
} \
|
|
}
|
|
CHIP_io_read_multi_N(1,uint8_t)
|
|
CHIP_io_read_multi_N(2,uint16_t)
|
|
CHIP_io_read_multi_N(4,uint32_t)
|
|
CHIP_io_read_multi_N(8,uint64_t)
|
|
|
|
#define CHIP_io_read_region_N(BYTES,TYPE) \
|
|
void \
|
|
__C(__C(CHIP,_io_read_region_),BYTES)(v, h, o, a, c) \
|
|
void *v; \
|
|
bus_space_handle_t h; \
|
|
bus_size_t o, c; \
|
|
TYPE *a; \
|
|
{ \
|
|
\
|
|
while (c-- > 0) { \
|
|
*a++ = __C(__C(CHIP,_io_read_),BYTES)(v, h, o); \
|
|
o += sizeof *a; \
|
|
} \
|
|
}
|
|
CHIP_io_read_region_N(1,uint8_t)
|
|
CHIP_io_read_region_N(2,uint16_t)
|
|
CHIP_io_read_region_N(4,uint32_t)
|
|
CHIP_io_read_region_N(8,uint64_t)
|
|
|
|
inline void
|
|
__C(CHIP,_io_write_1)(v, ioh, off, val)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t off;
|
|
uint8_t val;
|
|
{
|
|
uint8_t *ptr = (void *)(ioh + (off << CHIP_ALIGN_STRIDE));
|
|
|
|
*ptr = val;
|
|
}
|
|
|
|
inline void
|
|
__C(CHIP,_io_write_2)(v, ioh, off, val)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t off;
|
|
uint16_t val;
|
|
{
|
|
#if CHIP_ALIGN_STRIDE >= 1
|
|
uint16_t *ptr = (void *)(ioh + (off << (CHIP_ALIGN_STRIDE - 1)));
|
|
#else
|
|
uint16_t *ptr = (void *)(ioh + off);
|
|
#endif
|
|
|
|
*ptr = val;
|
|
}
|
|
|
|
inline void
|
|
__C(CHIP,_io_write_4)(v, ioh, off, val)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t off;
|
|
uint32_t val;
|
|
{
|
|
#if CHIP_ALIGN_STRIDE >= 2
|
|
uint32_t *ptr = (void *)(ioh + (off << (CHIP_ALIGN_STRIDE - 2)));
|
|
#else
|
|
uint32_t *ptr = (void *)(ioh + off);
|
|
#endif
|
|
|
|
*ptr = val;
|
|
}
|
|
|
|
inline void
|
|
__C(CHIP,_io_write_8)(v, ioh, off, val)
|
|
void *v;
|
|
bus_space_handle_t ioh;
|
|
bus_size_t off;
|
|
uint64_t val;
|
|
{
|
|
|
|
/* XXX XXX XXX */
|
|
panic("%s not implemented", __S(__C(CHIP,_io_write_8)));
|
|
}
|
|
|
|
#define CHIP_io_write_multi_N(BYTES,TYPE) \
|
|
void \
|
|
__C(__C(CHIP,_io_write_multi_),BYTES)(v, h, o, a, c) \
|
|
void *v; \
|
|
bus_space_handle_t h; \
|
|
bus_size_t o, c; \
|
|
const TYPE *a; \
|
|
{ \
|
|
\
|
|
while (c-- > 0) { \
|
|
__C(__C(CHIP,_io_write_),BYTES)(v, h, o, *a++); \
|
|
__C(CHIP,_io_barrier)(v, h, o, sizeof *a, \
|
|
BUS_SPACE_BARRIER_WRITE); \
|
|
} \
|
|
}
|
|
CHIP_io_write_multi_N(1,uint8_t)
|
|
CHIP_io_write_multi_N(2,uint16_t)
|
|
CHIP_io_write_multi_N(4,uint32_t)
|
|
CHIP_io_write_multi_N(8,uint64_t)
|
|
|
|
#define CHIP_io_write_region_N(BYTES,TYPE) \
|
|
void \
|
|
__C(__C(CHIP,_io_write_region_),BYTES)(v, h, o, a, c) \
|
|
void *v; \
|
|
bus_space_handle_t h; \
|
|
bus_size_t o, c; \
|
|
const TYPE *a; \
|
|
{ \
|
|
\
|
|
while (c-- > 0) { \
|
|
__C(__C(CHIP,_io_write_),BYTES)(v, h, o, *a++); \
|
|
o += sizeof *a; \
|
|
} \
|
|
}
|
|
CHIP_io_write_region_N(1,uint8_t)
|
|
CHIP_io_write_region_N(2,uint16_t)
|
|
CHIP_io_write_region_N(4,uint32_t)
|
|
CHIP_io_write_region_N(8,uint64_t)
|
|
|
|
#define CHIP_io_set_multi_N(BYTES,TYPE) \
|
|
void \
|
|
__C(__C(CHIP,_io_set_multi_),BYTES)(v, h, o, val, c) \
|
|
void *v; \
|
|
bus_space_handle_t h; \
|
|
bus_size_t o, c; \
|
|
TYPE val; \
|
|
{ \
|
|
\
|
|
while (c-- > 0) { \
|
|
__C(__C(CHIP,_io_write_),BYTES)(v, h, o, val); \
|
|
__C(CHIP,_io_barrier)(v, h, o, sizeof val, \
|
|
BUS_SPACE_BARRIER_WRITE); \
|
|
} \
|
|
}
|
|
CHIP_io_set_multi_N(1,uint8_t)
|
|
CHIP_io_set_multi_N(2,uint16_t)
|
|
CHIP_io_set_multi_N(4,uint32_t)
|
|
CHIP_io_set_multi_N(8,uint64_t)
|
|
|
|
#define CHIP_io_set_region_N(BYTES,TYPE) \
|
|
void \
|
|
__C(__C(CHIP,_io_set_region_),BYTES)(v, h, o, val, c) \
|
|
void *v; \
|
|
bus_space_handle_t h; \
|
|
bus_size_t o, c; \
|
|
TYPE val; \
|
|
{ \
|
|
\
|
|
while (c-- > 0) { \
|
|
__C(__C(CHIP,_io_write_),BYTES)(v, h, o, val); \
|
|
o += sizeof val; \
|
|
} \
|
|
}
|
|
CHIP_io_set_region_N(1,uint8_t)
|
|
CHIP_io_set_region_N(2,uint16_t)
|
|
CHIP_io_set_region_N(4,uint32_t)
|
|
CHIP_io_set_region_N(8,uint64_t)
|
|
|
|
#define CHIP_io_copy_region_N(BYTES) \
|
|
void \
|
|
__C(__C(CHIP,_io_copy_region_),BYTES)(v, h1, o1, h2, o2, c) \
|
|
void *v; \
|
|
bus_space_handle_t h1, h2; \
|
|
bus_size_t o1, o2, c; \
|
|
{ \
|
|
bus_size_t o; \
|
|
\
|
|
if ((h1 + o1) >= (h2 + o2)) { \
|
|
/* src after dest: copy forward */ \
|
|
for (o = 0; c != 0; c--, o += BYTES) \
|
|
__C(__C(CHIP,_io_write_),BYTES)(v, h2, o2 + o, \
|
|
__C(__C(CHIP,_io_read_),BYTES)(v, h1, o1 + o)); \
|
|
} else { \
|
|
/* dest after src: copy backwards */ \
|
|
for (o = (c - 1) * BYTES; c != 0; c--, o -= BYTES) \
|
|
__C(__C(CHIP,_io_write_),BYTES)(v, h2, o2 + o, \
|
|
__C(__C(CHIP,_io_read_),BYTES)(v, h1, o1 + o)); \
|
|
} \
|
|
}
|
|
CHIP_io_copy_region_N(1)
|
|
CHIP_io_copy_region_N(2)
|
|
CHIP_io_copy_region_N(4)
|
|
CHIP_io_copy_region_N(8)
|