[BACK]Return to mainbus.c CVS log [TXT][DIR] Up to [local] / sys / arch / hppa64 / hppa64

Annotation of sys/arch/hppa64/hppa64/mainbus.c, Revision 1.1.1.1

1.1       nbrk        1: /*     $OpenBSD: mainbus.c,v 1.4 2007/05/29 21:00:50 jason Exp $       */
                      2:
                      3: /*
                      4:  * Copyright (c) 2005 Michael Shalayeff
                      5:  * All rights reserved.
                      6:  *
                      7:  * Permission to use, copy, modify, and distribute this software for any
                      8:  * purpose with or without fee is hereby granted, provided that the above
                      9:  * copyright notice and this permission notice appear in all copies.
                     10:  *
                     11:  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
                     12:  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
                     13:  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
                     14:  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
                     15:  * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER IN
                     16:  * AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
                     17:  * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
                     18:  */
                     19:
                     20: #include "power.h"
                     21:
                     22: #undef BTLBDEBUG
                     23:
                     24: #include <sys/param.h>
                     25: #include <sys/systm.h>
                     26: #include <sys/device.h>
                     27: #include <sys/reboot.h>
                     28: #include <sys/extent.h>
                     29: #include <sys/mbuf.h>
                     30:
                     31: #include <uvm/uvm.h>
                     32: #include <uvm/uvm_page.h>
                     33:
                     34: #include <machine/pdc.h>
                     35: #include <machine/iomod.h>
                     36: #include <machine/autoconf.h>
                     37:
                     38: struct mainbus_softc {
                     39:        struct  device sc_dv;
                     40:
                     41:        hppa_hpa_t sc_hpa;
                     42: };
                     43:
                     44: int    mbmatch(struct device *, void *, void *);
                     45: void   mbattach(struct device *, struct device *, void *);
                     46:
                     47: struct cfattach mainbus_ca = {
                     48:        sizeof(struct mainbus_softc), mbmatch, mbattach
                     49: };
                     50:
                     51: struct cfdriver mainbus_cd = {
                     52:        NULL, "mainbus", DV_DULL
                     53: };
                     54:
                     55: struct pdc_hpa pdc_hpa PDC_ALIGNMENT;
                     56: struct pdc_power_info pdc_power_info PDC_ALIGNMENT;
                     57:
                     58: /* from machdep.c */
                     59: extern struct extent *hppa_ex;
                     60:
                     61: int
                     62: mbus_add_mapping(bus_addr_t bpa, bus_size_t size, int flags,
                     63:     bus_space_handle_t *bshp)
                     64: {
                     65:        paddr_t spa, epa;
                     66:        int bank, off;
                     67:
                     68:        if ((bank = vm_physseg_find(atop(bpa), &off)) >= 0)
                     69:                panic("mbus_add_mapping: mapping real memory @0x%lx", bpa);
                     70:
                     71:        for (spa = trunc_page(bpa), epa = bpa + size;
                     72:             spa < epa; spa += PAGE_SIZE)
                     73:                pmap_kenter_pa(spa, spa, UVM_PROT_RW);
                     74:
                     75:        *bshp = bpa;
                     76:        return (0);
                     77: }
                     78:
                     79: int
                     80: mbus_map(void *v, bus_addr_t bpa, bus_size_t size,
                     81:     int flags, bus_space_handle_t *bshp)
                     82: {
                     83:        int error;
                     84:
                     85:        bpa &= HPPA_PHYSMAP;
                     86:        if ((error = extent_alloc_region(hppa_ex, bpa, size, EX_NOWAIT)))
                     87:                return (error);
                     88:
                     89:        if ((error = mbus_add_mapping(bpa, size, flags, bshp))) {
                     90:                if (extent_free(hppa_ex, bpa, size, EX_NOWAIT)) {
                     91:                        printf("bus_space_map: pa 0x%lx, size 0x%lx\n",
                     92:                                bpa, size);
                     93:                        printf("bus_space_map: can't free region\n");
                     94:                }
                     95:        }
                     96:
                     97:        return error;
                     98: }
                     99:
                    100: void
                    101: mbus_unmap(void *v, bus_space_handle_t bsh, bus_size_t size)
                    102: {
                    103:        u_long sva, eva;
                    104:
                    105:        sva = trunc_page(bsh);
                    106:        eva = round_page(bsh + size);
                    107:
                    108: #ifdef DIAGNOSTIC
                    109:        if (eva <= sva)
                    110:                panic("bus_space_unmap: overflow");
                    111: #endif
                    112:
                    113:        pmap_kremove(sva, eva - sva);
                    114:
                    115:        if (extent_free(hppa_ex, bsh, size, EX_NOWAIT)) {
                    116:                printf("bus_space_unmap: ps 0x%lx, size 0x%lx\n",
                    117:                    bsh, size);
                    118:                printf("bus_space_unmap: can't free region\n");
                    119:        }
                    120: }
                    121:
                    122: int
                    123: mbus_alloc(void *v, bus_addr_t rstart, bus_addr_t rend, bus_size_t size,
                    124:         bus_size_t align, bus_size_t boundary, int flags,
                    125:         bus_addr_t *addrp, bus_space_handle_t *bshp)
                    126: {
                    127:        bus_addr_t bpa;
                    128:        int error;
                    129:
                    130:        rstart &= HPPA_PHYSMAP;
                    131:        rend &= HPPA_PHYSMAP;
                    132:        if (rstart < hppa_ex->ex_start || rend > hppa_ex->ex_end)
                    133:                panic("bus_space_alloc: bad region start/end");
                    134:
                    135:        if ((error = extent_alloc_subregion(hppa_ex, rstart, rend, size,
                    136:            align, 0, boundary, EX_NOWAIT, &bpa)))
                    137:                return (error);
                    138:
                    139:        if ((error = mbus_add_mapping(bpa, size, flags, bshp))) {
                    140:                if (extent_free(hppa_ex, bpa, size, EX_NOWAIT)) {
                    141:                        printf("bus_space_alloc: pa 0x%lx, size 0x%lx\n",
                    142:                                bpa, size);
                    143:                        printf("bus_space_alloc: can't free region\n");
                    144:                }
                    145:        }
                    146:
                    147:        *addrp = bpa | ~HPPA_PHYSMAP;
                    148:        return (error);
                    149: }
                    150:
                    151: void
                    152: mbus_free(void *v, bus_space_handle_t h, bus_size_t size)
                    153: {
                    154:        /* bus_space_unmap() does all that we need to do. */
                    155:        mbus_unmap(v, h, size);
                    156: }
                    157:
                    158: int
                    159: mbus_subregion(void *v, bus_space_handle_t bsh, bus_size_t offset,
                    160:     bus_size_t size, bus_space_handle_t *nbshp)
                    161: {
                    162:        *nbshp = bsh + offset;
                    163:        return (0);
                    164: }
                    165:
                    166: void
                    167: mbus_barrier(void *v, bus_space_handle_t h, bus_size_t o, bus_size_t l, int op)
                    168: {
                    169:        sync_caches();
                    170: }
                    171:
                    172: struct hppa64_bus_space_tag hppa_bustag = {
                    173:        NULL,
                    174:
                    175:        mbus_map, mbus_unmap, mbus_subregion, mbus_alloc, mbus_free,
                    176:        mbus_barrier,
                    177: };
                    178:
                    179: int
                    180: mbus_dmamap_create(void *v, bus_size_t size, int nsegments,
                    181:                   bus_size_t maxsegsz, bus_size_t boundary, int flags,
                    182:                   bus_dmamap_t *dmamp)
                    183: {
                    184:        struct hppa64_bus_dmamap *map;
                    185:        size_t mapsize;
                    186:
                    187:        mapsize = sizeof(*map) + (sizeof(bus_dma_segment_t) * (nsegments - 1));
                    188:        map = malloc(mapsize, M_DEVBUF,
                    189:                (flags & BUS_DMA_NOWAIT) ? M_NOWAIT : M_WAITOK);
                    190:        if (!map)
                    191:                return (ENOMEM);
                    192:
                    193:        bzero(map, mapsize);
                    194:        map->_dm_size = size;
                    195:        map->_dm_segcnt = nsegments;
                    196:        map->_dm_maxsegsz = maxsegsz;
                    197:        map->_dm_boundary = boundary;
                    198:        map->_dm_flags = flags & ~(BUS_DMA_WAITOK|BUS_DMA_NOWAIT);
                    199:        map->dm_mapsize = 0;
                    200:        map->dm_nsegs = 0;
                    201:
                    202:        *dmamp = map;
                    203:        return (0);
                    204: }
                    205:
                    206: void
                    207: mbus_dmamap_unload(void *v, bus_dmamap_t map)
                    208: {
                    209:        map->dm_mapsize = 0;
                    210:        map->dm_nsegs = 0;
                    211: }
                    212:
                    213: void
                    214: mbus_dmamap_destroy(void *v, bus_dmamap_t map)
                    215: {
                    216:        if (map->dm_mapsize != 0)
                    217:                mbus_dmamap_unload(v, map);
                    218:
                    219:        free(map, M_DEVBUF);
                    220: }
                    221:
                    222: /*
                    223:  * Utility function to load a linear buffer.  lastaddrp holds state
                    224:  * between invocations (for multiple-buffer loads).  segp contains
                    225:  * the starting segment on entrance, and the ending segment on exit.
                    226:  * first indicates if this is the first invocation of this function.
                    227:  */
                    228: int
                    229: _bus_dmamap_load_buffer(bus_dma_tag_t t, bus_dmamap_t map, void *buf,
                    230:     bus_size_t buflen, struct proc *p, int flags, paddr_t *lastaddrp,
                    231:     int *segp, int first)
                    232: {
                    233:        bus_size_t sgsize;
                    234:        bus_addr_t curaddr, lastaddr, baddr, bmask;
                    235:        vaddr_t vaddr = (vaddr_t)buf;
                    236:        int seg;
                    237:        pmap_t pmap;
                    238:
                    239:        pmap = p? p->p_vmspace->vm_map.pmap : pmap_kernel();
                    240:        lastaddr = *lastaddrp;
                    241:        bmask  = ~(map->_dm_boundary - 1);
                    242:
                    243:        for (seg = *segp; buflen > 0 ; ) {
                    244:                /*
                    245:                 * Get the physical address for this segment.
                    246:                 */
                    247:                pmap_extract(pmap, vaddr, (paddr_t *)&curaddr);
                    248:
                    249:                /*
                    250:                 * Compute the segment size, and adjust counts.
                    251:                 */
                    252:                sgsize = PAGE_SIZE - ((u_long)vaddr & PAGE_MASK);
                    253:                if (buflen < sgsize)
                    254:                        sgsize = buflen;
                    255:
                    256:                /*
                    257:                 * Make sure we don't cross any boundaries.
                    258:                 */
                    259:                if (map->_dm_boundary > 0) {
                    260:                        baddr = (curaddr + map->_dm_boundary) & bmask;
                    261:                        if (sgsize > (baddr - curaddr))
                    262:                                sgsize = (baddr - curaddr);
                    263:                }
                    264:
                    265:                /*
                    266:                 * Insert chunk into a segment, coalescing with
                    267:                 * previous segment if possible.
                    268:                 */
                    269:                if (first) {
                    270:                        map->dm_segs[seg].ds_addr = curaddr;
                    271:                        map->dm_segs[seg].ds_len = sgsize;
                    272:                        map->dm_segs[seg]._ds_va = vaddr;
                    273:                        first = 0;
                    274:                } else {
                    275:                        if (curaddr == lastaddr &&
                    276:                            (map->dm_segs[seg].ds_len + sgsize) <=
                    277:                             map->_dm_maxsegsz &&
                    278:                            (map->_dm_boundary == 0 ||
                    279:                             (map->dm_segs[seg].ds_addr & bmask) ==
                    280:                             (curaddr & bmask)))
                    281:                                map->dm_segs[seg].ds_len += sgsize;
                    282:                        else {
                    283:                                if (++seg >= map->_dm_segcnt)
                    284:                                        break;
                    285:                                map->dm_segs[seg].ds_addr = curaddr;
                    286:                                map->dm_segs[seg].ds_len = sgsize;
                    287:                                map->dm_segs[seg]._ds_va = vaddr;
                    288:                        }
                    289:                }
                    290:
                    291:                lastaddr = curaddr + sgsize;
                    292:                vaddr += sgsize;
                    293:                buflen -= sgsize;
                    294:        }
                    295:
                    296:        *segp = seg;
                    297:        *lastaddrp = lastaddr;
                    298:
                    299:        /*
                    300:         * Did we fit?
                    301:         */
                    302:        if (buflen != 0)
                    303:                return (EFBIG);         /* XXX better return value here? */
                    304:        return (0);
                    305: }
                    306:
                    307: int
                    308: mbus_dmamap_load(void *v, bus_dmamap_t map, void *addr, bus_size_t size,
                    309:                 struct proc *p, int flags)
                    310: {
                    311:        paddr_t lastaddr;
                    312:        int seg, error;
                    313:
                    314:        /*
                    315:         * Make sure that on error condition we return "no valid mappings".
                    316:         */
                    317:        map->dm_nsegs = 0;
                    318:        map->dm_mapsize = 0;
                    319:
                    320:        if (size > map->_dm_size)
                    321:                return (EINVAL);
                    322:
                    323:        seg = 0;
                    324:        lastaddr = 0;
                    325:        error = _bus_dmamap_load_buffer(NULL, map, addr, size, p, flags,
                    326:            &lastaddr, &seg, 1);
                    327:        if (error == 0) {
                    328:                map->dm_mapsize = size;
                    329:                map->dm_nsegs = seg + 1;
                    330:        }
                    331:
                    332:        return (0);
                    333: }
                    334:
                    335: int
                    336: mbus_dmamap_load_mbuf(void *v, bus_dmamap_t map, struct mbuf *m0, int flags)
                    337: {
                    338:        paddr_t lastaddr;
                    339:        int seg, error, first;
                    340:        struct mbuf *m;
                    341:
                    342:        map->dm_mapsize = 0;
                    343:        map->dm_nsegs = 0;
                    344:
                    345: #ifdef DIAGNOSTIC
                    346:        if ((m0->m_flags & M_PKTHDR) == 0)
                    347:                panic("_bus_dmamap_load_mbuf: no packet header");
                    348: #endif
                    349:
                    350:        if (m0->m_pkthdr.len > map->_dm_size)
                    351:                return (EINVAL);
                    352:
                    353:        first = 1;
                    354:        seg = 0;
                    355:        error = 0;
                    356:        lastaddr = 0;
                    357:        for (m = m0; m != NULL && error == 0; m = m->m_next) {
                    358:                error = _bus_dmamap_load_buffer(NULL, map, m->m_data, m->m_len,
                    359:                    NULL, flags, &lastaddr, &seg, first);
                    360:                first = 0;
                    361:        }
                    362:        if (error == 0) {
                    363:                map->dm_mapsize = m0->m_pkthdr.len;
                    364:                map->dm_nsegs = seg + 1;
                    365:        }
                    366:
                    367:        return (error);
                    368: }
                    369:
                    370: int
                    371: mbus_dmamap_load_uio(void *v, bus_dmamap_t map, struct uio *uio, int flags)
                    372: {
                    373:        paddr_t lastaddr;
                    374:        int seg, i, error, first;
                    375:        bus_size_t minlen, resid;
                    376:        struct proc *p = NULL;
                    377:        struct iovec *iov;
                    378:        caddr_t addr;
                    379:
                    380:        /*
                    381:         * Make sure that on error condition we return "no valid mappings".
                    382:         */
                    383:        map->dm_mapsize = 0;
                    384:        map->dm_nsegs = 0;
                    385:
                    386:        resid = uio->uio_resid;
                    387:        iov = uio->uio_iov;
                    388:
                    389:        if (resid > map->_dm_size)
                    390:                return (EINVAL);
                    391:
                    392:        if (uio->uio_segflg == UIO_USERSPACE) {
                    393:                p = uio->uio_procp;
                    394: #ifdef DIAGNOSTIC
                    395:                if (p == NULL)
                    396:                        panic("_bus_dmamap_load_uio: USERSPACE but no proc");
                    397: #endif
                    398:        }
                    399:
                    400:        first = 1;
                    401:        seg = 0;
                    402:        error = 0;
                    403:        lastaddr = 0;
                    404:        for (i = 0; i < uio->uio_iovcnt && resid != 0 && error == 0; i++) {
                    405:                /*
                    406:                 * Now at the first iovec to load.  Load each iovec
                    407:                 * until we have exhausted the residual count.
                    408:                 */
                    409:                minlen = resid < iov[i].iov_len ? resid : iov[i].iov_len;
                    410:                addr = (caddr_t)iov[i].iov_base;
                    411:
                    412:                error = _bus_dmamap_load_buffer(NULL, map, addr, minlen,
                    413:                    p, flags, &lastaddr, &seg, first);
                    414:                first = 0;
                    415:
                    416:                resid -= minlen;
                    417:        }
                    418:        if (error == 0) {
                    419:                map->dm_mapsize = uio->uio_resid;
                    420:                map->dm_nsegs = seg + 1;
                    421:        }
                    422:        return (error);
                    423: }
                    424:
                    425: int
                    426: mbus_dmamap_load_raw(void *v, bus_dmamap_t map, bus_dma_segment_t *segs,
                    427:     int nsegs, bus_size_t size, int flags)
                    428: {
                    429:        if (nsegs > map->_dm_segcnt || size > map->_dm_size)
                    430:                return (EINVAL);
                    431:
                    432:        /*
                    433:         * Make sure we don't cross any boundaries.
                    434:         */
                    435:        if (map->_dm_boundary) {
                    436:                bus_addr_t bmask = ~(map->_dm_boundary - 1);
                    437:                int i;
                    438:
                    439:                for (i = 0; i < nsegs; i++) {
                    440:                        if (segs[i].ds_len > map->_dm_maxsegsz)
                    441:                                return (EINVAL);
                    442:                        if ((segs[i].ds_addr & bmask) !=
                    443:                            ((segs[i].ds_addr + segs[i].ds_len - 1) & bmask))
                    444:                                return (EINVAL);
                    445:                }
                    446:        }
                    447:
                    448:        bcopy(segs, map->dm_segs, nsegs * sizeof(*segs));
                    449:        map->dm_nsegs = nsegs;
                    450:        map->dm_mapsize = size;
                    451:        return (0);
                    452: }
                    453:
                    454: void
                    455: mbus_dmamap_sync(void *v, bus_dmamap_t map, bus_addr_t off, bus_size_t len,
                    456:     int ops)
                    457: {
                    458:        bus_dma_segment_t *ps = map->dm_segs,
                    459:            *es = &map->dm_segs[map->dm_nsegs];
                    460:
                    461:        if (off >= map->_dm_size)
                    462:                return;
                    463:
                    464:        if ((off + len) > map->_dm_size)
                    465:                len = map->_dm_size - off;
                    466:
                    467:        for (; len && ps < es; ps++)
                    468:                if (off > ps->ds_len)
                    469:                        off -= ps->ds_len;
                    470:                else {
                    471:                        bus_size_t l = ps->ds_len - off;
                    472:                        if (l > len)
                    473:                                l = len;
                    474:                        fdcache(HPPA_SID_KERNEL, ps->_ds_va + off, l);
                    475:                        len -= l;
                    476:                        off = 0;
                    477:                }
                    478:
                    479:        /* for either operation sync the shit away */
                    480:        sync_caches();
                    481: }
                    482:
                    483: int
                    484: mbus_dmamem_alloc(void *v, bus_size_t size, bus_size_t alignment,
                    485:                  bus_size_t boundary, bus_dma_segment_t *segs, int nsegs,
                    486:                  int *rsegs, int flags)
                    487: {
                    488:        extern paddr_t avail_end;
                    489:        struct pglist pglist;
                    490:        struct vm_page *pg;
                    491:
                    492:        size = round_page(size);
                    493:
                    494:        TAILQ_INIT(&pglist);
                    495:        if (uvm_pglistalloc(size, 0, avail_end, alignment, boundary,
                    496:            &pglist, 1, flags & BUS_DMA_NOWAIT))
                    497:                return (ENOMEM);
                    498:
                    499:        pg = TAILQ_FIRST(&pglist);
                    500:        segs[0]._ds_va = segs[0].ds_addr = VM_PAGE_TO_PHYS(pg);
                    501:        segs[0].ds_len = size;
                    502:        *rsegs = 1;
                    503:
                    504:        for(; pg; pg = TAILQ_NEXT(pg, pageq))
                    505:                /* XXX for now */
                    506:                pmap_changebit(pg, PTE_UNCACHABLE, 0);
                    507:        pmap_update(pmap_kernel());
                    508:
                    509:        return (0);
                    510: }
                    511:
                    512: void
                    513: mbus_dmamem_free(void *v, bus_dma_segment_t *segs, int nsegs)
                    514: {
                    515:        struct pglist pglist;
                    516:        paddr_t pa, epa;
                    517:
                    518:        TAILQ_INIT(&pglist);
                    519:        for(; nsegs--; segs++)
                    520:                for (pa = segs->ds_addr, epa = pa + segs->ds_len;
                    521:                     pa < epa; pa += PAGE_SIZE) {
                    522:                        struct vm_page *pg = PHYS_TO_VM_PAGE(pa);
                    523:                        if (!pg)
                    524:                                panic("mbus_dmamem_free: no page for pa");
                    525:                        TAILQ_INSERT_TAIL(&pglist, pg, pageq);
                    526:                }
                    527:        uvm_pglistfree(&pglist);
                    528: }
                    529:
                    530: int
                    531: mbus_dmamem_map(void *v, bus_dma_segment_t *segs, int nsegs, size_t size,
                    532:                caddr_t *kvap, int flags)
                    533: {
                    534:        *kvap = (caddr_t)segs[0].ds_addr;
                    535:        return 0;
                    536: }
                    537:
                    538: void
                    539: mbus_dmamem_unmap(void *v, caddr_t kva, size_t size)
                    540: {
                    541: }
                    542:
                    543: paddr_t
                    544: mbus_dmamem_mmap(void *v, bus_dma_segment_t *segs, int nsegs, off_t off,
                    545:                 int prot, int flags)
                    546: {
                    547:        panic("_dmamem_mmap: not implemented");
                    548: }
                    549:
                    550: const struct hppa64_bus_dma_tag hppa_dmatag = {
                    551:        NULL,
                    552:        mbus_dmamap_create, mbus_dmamap_destroy,
                    553:        mbus_dmamap_load, mbus_dmamap_load_mbuf,
                    554:        mbus_dmamap_load_uio, mbus_dmamap_load_raw,
                    555:        mbus_dmamap_unload, mbus_dmamap_sync,
                    556:
                    557:        mbus_dmamem_alloc, mbus_dmamem_free, mbus_dmamem_map,
                    558:        mbus_dmamem_unmap, mbus_dmamem_mmap
                    559: };
                    560:
                    561: int
                    562: mbmatch(parent, cfdata, aux)
                    563:        struct device *parent;
                    564:        void *cfdata;
                    565:        void *aux;
                    566: {
                    567:        struct cfdata *cf = cfdata;
                    568:
                    569:        /* there will be only one */
                    570:        if (cf->cf_unit)
                    571:                return 0;
                    572:
                    573:        return 1;
                    574: }
                    575:
                    576: void
                    577: mbattach(parent, self, aux)
                    578:        struct device *parent;
                    579:        struct device *self;
                    580:        void *aux;
                    581: {
                    582:        struct mainbus_softc *sc = (struct mainbus_softc *)self;
                    583:        struct confargs nca;
                    584:        bus_space_handle_t ioh;
                    585:        bus_addr_t hpa;
                    586:
                    587:        /* fetch the "default" cpu hpa */
                    588:        if (pdc_call((iodcio_t)pdc, 0, PDC_HPA, PDC_HPA_DFLT, &pdc_hpa) < 0)
                    589:                panic("mbattach: PDC_HPA failed");
                    590:        hpa = pdc_hpa.hpa | 0xffffffff00000000UL;
                    591:
                    592:        printf(" [flex %lx]\n", hpa & HPPA_FLEX_MASK);
                    593:
                    594:        /* map all the way till the end of the memory */
                    595:        if (bus_space_map(&hppa_bustag, hpa, HPPA_PHYSEND - hpa + 1, 0, &ioh))
                    596:                panic("mbattach: cannot map mainbus IO space");
                    597:
                    598:        /*
                    599:         * Local-Broadcast the HPA to all modules on this bus
                    600:         */
                    601:        ((struct iomod *)(HPPA_LBCAST & HPPA_PHYSMAP))->io_flex =
                    602:            (hpa & HPPA_FLEX_MASK) | DMA_ENABLE;
                    603:
                    604:        sc->sc_hpa = hpa;
                    605:
                    606:        /* PDC first */
                    607:        bzero (&nca, sizeof(nca));
                    608:        nca.ca_name = "pdc";
                    609:        nca.ca_iot = &hppa_bustag;
                    610:        nca.ca_dmatag = &hppa_dmatag;
                    611:        config_found(self, &nca, mbprint);
                    612:
                    613: #if NPOWER > 0
                    614:        /* get some power */
                    615:        bzero (&nca, sizeof(nca));
                    616:        nca.ca_name = "power";
                    617:        nca.ca_irq = -1;
                    618:        if (!pdc_call((iodcio_t)pdc, 0, PDC_SOFT_POWER,
                    619:            PDC_SOFT_POWER_INFO, &pdc_power_info, 0)) {
                    620:                nca.ca_iot = &hppa_bustag;
                    621:                nca.ca_hpa = pdc_power_info.addr;
                    622:                nca.ca_hpamask = HPPA_IOBEGIN;
                    623:        }
                    624:        config_found(self, &nca, mbprint);
                    625: #endif
                    626:
                    627:        bzero (&nca, sizeof(nca));
                    628:        nca.ca_iot = &hppa_bustag;
                    629:        nca.ca_dmatag = &hppa_dmatag;
                    630:        nca.ca_mod = -1;
                    631:        pdc_scan(self, &nca);
                    632: }
                    633:
                    634: /*
                    635:  * retrive CPU #N HPA value
                    636:  */
                    637: hppa_hpa_t
                    638: cpu_gethpa(n)
                    639:        int n;
                    640: {
                    641:        struct mainbus_softc *sc;
                    642:
                    643:        sc = mainbus_cd.cd_devs[0];
                    644:
                    645:        return sc->sc_hpa;
                    646: }
                    647:
                    648: int
                    649: mbprint(aux, pnp)
                    650:        void *aux;
                    651:        const char *pnp;
                    652: {
                    653:        struct confargs *ca = aux;
                    654:
                    655:        if (pnp)
                    656:                printf("\"%s\" at %s", ca->ca_name, pnp);
                    657:
                    658:        if (ca->ca_hpa)
                    659:                printf(" hpa %lx", ca->ca_hpa);
                    660:
                    661:        return (UNCONF);
                    662: }

CVSweb