root/sys/dev/ic/ufshcivar.h
/*      $OpenBSD: ufshcivar.h,v 1.12 2025/01/18 19:42:39 mglocker Exp $ */

/*
 * Copyright (c) 2022 Marcus Glocker <mglocker@openbsd.org>
 *
 * Permission to use, copy, modify, and distribute this software for any
 * purpose with or without fee is hereby granted, provided that the above
 * copyright notice and this permission notice appear in all copies.
 *
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 */

/* #define UFSHCI_DEBUG */
#ifdef UFSHCI_DEBUG
extern int ufshci_debug;
#define DPRINTF(l, x...)        do { if ((l) <= ufshci_debug) printf(x); } \
                                    while (0)
#else
#define DPRINTF(l, x...)
#endif

#define UFSHCI_READ_4(sc, x) \
    bus_space_read_4((sc)->sc_iot, (sc)->sc_ioh, (x))
#define UFSHCI_WRITE_4(sc, x, y) \
    bus_space_write_4((sc)->sc_iot, (sc)->sc_ioh, (x), (y))

#define UFSHCI_DMA_MAP(_udm)    ((_udm)->udm_map)
#define UFSHCI_DMA_LEN(_udm)    ((_udm)->udm_map->dm_segs[0].ds_len)
#define UFSHCI_DMA_DVA(_udm)    ((uint64_t)(_udm)->udm_map->dm_segs[0].ds_addr)
#define UFSHCI_DMA_KVA(_udm)    ((void *)(_udm)->udm_kva)
struct ufshci_dmamem {
        bus_dmamap_t            udm_map;
        bus_dma_segment_t       udm_seg;
        size_t                  udm_size;
        caddr_t                 udm_kva;
};

struct ufshci_softc;

/* SCSI */
enum ccb_status {
        CCB_STATUS_FREE,
        CCB_STATUS_INPROGRESS,
        CCB_STATUS_READY2FREE,
        CCB_STATUS_COUNT
};
struct ufshci_ccb {
        SIMPLEQ_ENTRY(ufshci_ccb)        ccb_entry;
        bus_dmamap_t                     ccb_dmamap;
        void                            *ccb_cookie;
        int                              ccb_slot;
        int                              ccb_status;
        void                             (*ccb_done)(struct ufshci_softc *,
                                             struct ufshci_ccb *);
};
SIMPLEQ_HEAD(ufshci_ccb_list, ufshci_ccb);

struct ufshci_softc {
        struct device            sc_dev;

        bus_space_tag_t          sc_iot;
        bus_space_handle_t       sc_ioh;
        bus_size_t               sc_ios;
        bus_dma_tag_t            sc_dmat;

        uint8_t                  sc_iacth;
        struct mutex             sc_cmd_mtx;

#define UFSHCI_FLAGS_AGGR_INTR   1
        uint8_t                  sc_flags;
        uint32_t                 sc_ver;
        uint32_t                 sc_cap;
        uint32_t                 sc_hcpid;
        uint32_t                 sc_hcmid;
        uint8_t                  sc_nutmrs;
        uint8_t                  sc_rtt;
        uint8_t                  sc_nutrs;

        struct ufshci_dmamem    *sc_dmamem_utmrd;
        struct ufshci_dmamem    *sc_dmamem_utrd;
        struct ufshci_dmamem    *sc_dmamem_ucd;

        /* SCSI */
        struct scsi_iopool       sc_iopool;
        struct mutex             sc_ccb_mtx;
        struct ufshci_ccb_list   sc_ccb_list;
        struct ufshci_ccb       *sc_ccbs;

        /* kstat */
        uint64_t                *sc_stats_slots;
        struct mutex             sc_kstat_mtx_ccb;
        struct mutex             sc_kstat_mtx_slot;
        struct kstat            *sc_kstat_ccb;
        struct kstat            *sc_kstat_slot;
};

int     ufshci_intr(void *);
int     ufshci_attach(struct ufshci_softc *);
int     ufshci_activate(struct device *, int);