📄 aic7xxx_inline.h
字号:
/* * Inline routines shareable across OS platforms. * * Copyright (c) 1994-2001 Justin T. Gibbs. * Copyright (c) 2000-2001 Adaptec Inc. * All rights reserved. * * 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, * without modification. * 2. Redistributions in binary form must reproduce at minimum a disclaimer * substantially similar to the "NO WARRANTY" disclaimer below * ("Disclaimer") and any redistribution must be conditioned upon * including a substantially similar Disclaimer requirement for further * binary redistribution. * 3. Neither the names of the above-listed copyright holders nor the names * of any contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * Alternatively, this software may be distributed under the terms of the * GNU General Public License ("GPL") version 2 as published by the Free * Software Foundation. * * NO WARRANTY * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * HOLDERS OR CONTRIBUTORS BE LIABLE FOR 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 DAMAGES. * * $Id: //depot/aic7xxx/aic7xxx/aic7xxx_inline.h#43 $ * * $FreeBSD$ */#ifndef _AIC7XXX_INLINE_H_#define _AIC7XXX_INLINE_H_/************************* Sequencer Execution Control ************************/static __inline void ahc_pause_bug_fix(struct ahc_softc *ahc);static __inline int ahc_is_paused(struct ahc_softc *ahc);static __inline void ahc_pause(struct ahc_softc *ahc);static __inline void ahc_unpause(struct ahc_softc *ahc);/* * Work around any chip bugs related to halting sequencer execution. * On Ultra2 controllers, we must clear the CIOBUS stretch signal by * reading a register that will set this signal and deassert it. * Without this workaround, if the chip is paused, by an interrupt or * manual pause while accessing scb ram, accesses to certain registers * will hang the system (infinite pci retries). */static __inline voidahc_pause_bug_fix(struct ahc_softc *ahc){ if ((ahc->features & AHC_ULTRA2) != 0) (void)ahc_inb(ahc, CCSCBCTL);}/* * Determine whether the sequencer has halted code execution. * Returns non-zero status if the sequencer is stopped. */static __inline intahc_is_paused(struct ahc_softc *ahc){ return ((ahc_inb(ahc, HCNTRL) & PAUSE) != 0);}/* * Request that the sequencer stop and wait, indefinitely, for it * to stop. The sequencer will only acknowledge that it is paused * once it has reached an instruction boundary and PAUSEDIS is * cleared in the SEQCTL register. The sequencer may use PAUSEDIS * for critical sections. */static __inline voidahc_pause(struct ahc_softc *ahc){ ahc_outb(ahc, HCNTRL, ahc->pause); /* * Since the sequencer can disable pausing in a critical section, we * must loop until it actually stops. */ while (ahc_is_paused(ahc) == 0) ; ahc_pause_bug_fix(ahc);}/* * Allow the sequencer to continue program execution. * We check here to ensure that no additional interrupt * sources that would cause the sequencer to halt have been * asserted. If, for example, a SCSI bus reset is detected * while we are fielding a different, pausing, interrupt type, * we don't want to release the sequencer before going back * into our interrupt handler and dealing with this new * condition. */static __inline voidahc_unpause(struct ahc_softc *ahc){ if ((ahc_inb(ahc, INTSTAT) & (SCSIINT | SEQINT | BRKADRINT)) == 0) ahc_outb(ahc, HCNTRL, ahc->unpause);}/*********************** Untagged Transaction Routines ************************/static __inline void ahc_freeze_untagged_queues(struct ahc_softc *ahc);static __inline void ahc_release_untagged_queues(struct ahc_softc *ahc);/* * Block our completion routine from starting the next untagged * transaction for this target or target lun. */static __inline voidahc_freeze_untagged_queues(struct ahc_softc *ahc){ if ((ahc->flags & AHC_SCB_BTT) == 0) ahc->untagged_queue_lock++;}/* * Allow the next untagged transaction for this target or target lun * to be executed. We use a counting semaphore to allow the lock * to be acquired recursively. Once the count drops to zero, the * transaction queues will be run. */static __inline voidahc_release_untagged_queues(struct ahc_softc *ahc){ if ((ahc->flags & AHC_SCB_BTT) == 0) { ahc->untagged_queue_lock--; if (ahc->untagged_queue_lock == 0) ahc_run_untagged_queues(ahc); }}/************************** Memory mapping routines ***************************/static __inline struct ahc_dma_seg * ahc_sg_bus_to_virt(struct scb *scb, uint32_t sg_busaddr);static __inline uint32_t ahc_sg_virt_to_bus(struct scb *scb, struct ahc_dma_seg *sg);static __inline uint32_t ahc_hscb_busaddr(struct ahc_softc *ahc, u_int index);static __inline void ahc_sync_scb(struct ahc_softc *ahc, struct scb *scb, int op);static __inline void ahc_sync_sglist(struct ahc_softc *ahc, struct scb *scb, int op);static __inline uint32_t ahc_targetcmd_offset(struct ahc_softc *ahc, u_int index);static __inline struct ahc_dma_seg *ahc_sg_bus_to_virt(struct scb *scb, uint32_t sg_busaddr){ int sg_index; sg_index = (sg_busaddr - scb->sg_list_phys)/sizeof(struct ahc_dma_seg); /* sg_list_phys points to entry 1, not 0 */ sg_index++; return (&scb->sg_list[sg_index]);}static __inline uint32_tahc_sg_virt_to_bus(struct scb *scb, struct ahc_dma_seg *sg){ int sg_index; /* sg_list_phys points to entry 1, not 0 */ sg_index = sg - &scb->sg_list[1]; return (scb->sg_list_phys + (sg_index * sizeof(*scb->sg_list)));}static __inline uint32_tahc_hscb_busaddr(struct ahc_softc *ahc, u_int index){ return (ahc->scb_data->hscb_busaddr + (sizeof(struct hardware_scb) * index));}static __inline voidahc_sync_scb(struct ahc_softc *ahc, struct scb *scb, int op){ ahc_dmamap_sync(ahc, ahc->scb_data->hscb_dmat, ahc->scb_data->hscb_dmamap, /*offset*/(scb->hscb - ahc->hscbs) * sizeof(*scb->hscb), /*len*/sizeof(*scb->hscb), op);}static __inline voidahc_sync_sglist(struct ahc_softc *ahc, struct scb *scb, int op){ if (scb->sg_count == 0) return; ahc_dmamap_sync(ahc, ahc->scb_data->sg_dmat, scb->sg_map->sg_dmamap, /*offset*/(scb->sg_list - scb->sg_map->sg_vaddr) * sizeof(struct ahc_dma_seg), /*len*/sizeof(struct ahc_dma_seg) * scb->sg_count, op);}static __inline uint32_tahc_targetcmd_offset(struct ahc_softc *ahc, u_int index){ return (((uint8_t *)&ahc->targetcmds[index]) - ahc->qoutfifo);}/******************************** Debugging ***********************************/static __inline char *ahc_name(struct ahc_softc *ahc);static __inline char *ahc_name(struct ahc_softc *ahc){ return (ahc->name);}/*********************** Miscelaneous Support Functions ***********************/static __inline void ahc_update_residual(struct ahc_softc *ahc, struct scb *scb);static __inline struct ahc_initiator_tinfo * ahc_fetch_transinfo(struct ahc_softc *ahc, char channel, u_int our_id, u_int remote_id, struct ahc_tmode_tstate **tstate);static __inline uint16_t ahc_inw(struct ahc_softc *ahc, u_int port);static __inline void ahc_outw(struct ahc_softc *ahc, u_int port, u_int value);static __inline uint32_t ahc_inl(struct ahc_softc *ahc, u_int port);static __inline void ahc_outl(struct ahc_softc *ahc, u_int port, uint32_t value);static __inline uint64_t ahc_inq(struct ahc_softc *ahc, u_int port);static __inline void ahc_outq(struct ahc_softc *ahc, u_int port, uint64_t value);static __inline struct scb* ahc_get_scb(struct ahc_softc *ahc);static __inline void ahc_free_scb(struct ahc_softc *ahc, struct scb *scb);static __inline void ahc_swap_with_next_hscb(struct ahc_softc *ahc, struct scb *scb);static __inline void ahc_queue_scb(struct ahc_softc *ahc, struct scb *scb);static __inline struct scsi_sense_data * ahc_get_sense_buf(struct ahc_softc *ahc, struct scb *scb);static __inline uint32_t ahc_get_sense_bufaddr(struct ahc_softc *ahc, struct scb *scb);/* * Determine whether the sequencer reported a residual * for this SCB/transaction. */static __inline voidahc_update_residual(struct ahc_softc *ahc, struct scb *scb){ uint32_t sgptr; sgptr = ahc_le32toh(scb->hscb->sgptr); if ((sgptr & SG_RESID_VALID) != 0) ahc_calc_residual(ahc, scb);}/* * Return pointers to the transfer negotiation information * for the specified our_id/remote_id pair. */static __inline struct ahc_initiator_tinfo *ahc_fetch_transinfo(struct ahc_softc *ahc, char channel, u_int our_id, u_int remote_id, struct ahc_tmode_tstate **tstate){ /* * Transfer data structures are stored from the perspective * of the target role. Since the parameters for a connection * in the initiator role to a given target are the same as * when the roles are reversed, we pretend we are the target. */ if (channel == 'B') our_id += 8; *tstate = ahc->enabled_targets[our_id]; return (&(*tstate)->transinfo[remote_id]);}static __inline uint16_tahc_inw(struct ahc_softc *ahc, u_int port){ return ((ahc_inb(ahc, port+1) << 8) | ahc_inb(ahc, port));}static __inline voidahc_outw(struct ahc_softc *ahc, u_int port, u_int value){ ahc_outb(ahc, port, value & 0xFF); ahc_outb(ahc, port+1, (value >> 8) & 0xFF);}static __inline uint32_tahc_inl(struct ahc_softc *ahc, u_int port){ return ((ahc_inb(ahc, port)) | (ahc_inb(ahc, port+1) << 8) | (ahc_inb(ahc, port+2) << 16) | (ahc_inb(ahc, port+3) << 24));}static __inline voidahc_outl(struct ahc_softc *ahc, u_int port, uint32_t value){ ahc_outb(ahc, port, (value) & 0xFF);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -