/*	$NetBSD: octeon_usbcvar.h,v 1.6 2018/11/18 11:48:56 skrll Exp $	*/

/*
 * Copyright (c) 2007 Internet Initiative Japan, 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.
 * 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.
 *
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS 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 REGENTS 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.
 */

#ifndef _OCTEON_USBCVAR_H_
#define _OCTEON_USBCVAR_H_
#include <dev/usb/usb.h>

typedef u_int32_t octeon_usbc_physaddr_t;
typedef u_int32_t octeon_usbc_link_t;
struct octeon_usbc_soft_td;
struct octeon_usbc_soft_ed;
struct octeon_usbc_softc;

typedef enum octeon_usbc_halt_status{
	USBC_HALT_NO_HALT_STATUS,
	USBC_HALT_XACT_COMPLETE,
	USBC_HALT_ACK,
	USBC_HALT_NAK,
	USBC_HALT_NYET,
	USBC_HALT_STALL,
	USBC_HALT_XACT_ERR,
	USBC_HALT_FRAME_OVERRUN,
	USBC_HALT_CHANNEL_BABBLE_ERR,
	USBC_HALT_DATA_TOGGLE_ERR,
	USBC_HALT_AHB_ERR,
	USBC_HALT_PERIODIC_INCOMPLETE,
	USBC_HALT_XFER_COMPLETE
} octeon_usbc_halt_status_t;

#define USBC_DMA_BURST_SIZE 0

#define USBC_MAX_PERIO_FIFOS 15
#define USBC_MAX_ENDPOINTS_CHANNELS 16
typedef struct octeon_usbc_hostch { /* FIXME */
	u_int32_t	hc_id;
	u_int32_t	hc_num;

	struct octeon_usbc_softc *sc;
	callout_t	tmo_retry;

	struct octeon_usbc_soft_ed *sed;
	struct octeon_usbc_soft_td *std_cur;
	struct octeon_usbc_soft_td *std_done_head;

	u_int8_t	status;
#define USBC_HOST_CH_STATUS_FREE	0x0
#define USBC_HOST_CH_STATUS_RESERVED	0x1
#define USBC_HOST_CH_STATUS_WAIT_HLT	0x2
#define USBC_HOST_CH_STATUS_NOW_XFER	0x4
#define USBC_HOST_CH_STATUS_SLAVE_HALT	0x8

	u_int8_t	error_state	:1;

	/* stored registers */
	u_int32_t	hcint;
	u_int32_t	hcintmsk;
	u_int32_t	hcchar;
	u_int32_t	hcsplt;
	u_int32_t	hctsiz;

	/* register offsets */
	bus_size_t	offset_usbc_hcchar;
	bus_size_t	offset_usbc_hcsplt;
	bus_size_t	offset_usbc_hcint;
	bus_size_t	offset_usbc_hcintmsk;
	bus_size_t	offset_usbc_hctsiz;
#ifdef OCTEON_USBN_CN31XX_DMA_WORKAROUND
	bus_size_t	offset_usbc_dfifo;
#endif
} octeon_usbc_hostch_t;

typedef struct octeon_usbc_xfer {
	struct usbd_xfer xfer;
} octeon_usbc_xfer_t;


/* endpoint descriptor */
typedef struct octeon_usbc_soft_ed {
	struct octeon_usbc_soft_ed	*next;

	u_int8_t	ref_count;

	struct octeon_usbc_soft_td	*std_head;

	u_int8_t	status;
#define	USBC_ENDPT_STATUS_SKIP		0
#define USBC_ENDPT_STATUS_READY		1
#define USBC_ENDPT_STATUS_IN_PROGRESS	2
#define USBC_ENDPT_STATUS_WAIT_NEXT_SOF	3
#define USBC_ENDPT_STATUS_WAIT_FIFO_EMPTY	4
#define USBC_ENDPT_STATUS_WAIT_PING_ACK	5

	octeon_usbc_hostch_t	*hostch;

	u_int32_t	interval;
	u_int32_t	interval_count;
	u_int8_t	ping_state;
	u_int8_t	toggle_state;
#define USBC_DATA_TGL_DATA0		0x00
#define USBC_DATA_TGL_DATA1		0x01
#define USBC_DATA_TGL_DATA2		0x02
#define USBC_DATA_TGL_MDATA		0x03
#define USBC_DATA_TGL_SETUP		0x04
#define USBC_DATA_TGL_INHERIT		0xFF

	/* hcsplt params */
	u_int8_t	hcsplt_splt_ena;

	/* hcchar params */
	u_int16_t	hcchar_mps;
	u_int8_t	hcchar_epnum;
	u_int8_t	hcchar_devaddr;
	u_int8_t	hcchar_lspddev;
	u_int8_t	hcchar_eptype;
#define USBC_ENDPT_TYPE_CTRL	UE_CONTROL
#define USBC_ENDPT_TYPE_ISOC	UE_ISOCHRONOUS
#define USBC_ENDPT_TYPE_BULK	UE_BULK
#define USBC_ENDPT_TYPE_INTR	UE_INTERRUPT
#define USBC_ENDPT_TYPE_IS_PERIO(eptype) (((eptype) & 0x1) != 0x0)
} octeon_usbc_soft_ed_t;
#define USBC_SED_SIZE	( sizeof(octeon_usbc_soft_ed_t) )
#define USBC_SED_LIST_MAX 128

/* transaction descriptor */
typedef struct octeon_usbc_soft_td {
	struct octeon_usbc_soft_td	*next;
	struct octeon_usbc_soft_ed	*sed;
	usb_dma_t		*buff;
	struct usbd_xfer	*xfer;
	u_int32_t	offset;
	u_int32_t	base_offset;
	u_int8_t	error_count;
	u_int8_t	data_toggle;
	u_int8_t	direction;
#define USBC_ENDPT_DIRECTION_OUT	0
#define USBC_ENDPT_DIRECTION_IN		1
	u_int8_t  pid;
	u_int32_t len;
	u_int32_t actlen;
	u_int16_t flags;
#define USBC_STD_FLAG_CALL_DONE 0x0001
#define USBC_STD_FLAG_ADD_LEN	0x0002
#define USBC_STD_FLAG_SKIP	0x0004
	u_int8_t is_used;
} octeon_usbc_soft_td_t;
#define USBC_STD_SIZE	( sizeof(octeon_usbc_soft_td_t) )
#define USBC_STD_LIST_MAX 128


typedef struct octeon_usbc_softc {
	device_t		sc_dev;
	struct usbd_bus		sc_bus;

	bus_space_tag_t         sc_bust;	/* iobus space */
	bus_space_handle_t	sc_regh;	/* usbc register space */

	bus_size_t sc_size;


	usb_dma_t		sc_hccadma;
	u_int8_t		sc_dma_enable;

#ifdef OCTEON_USBN_CN31XX_DMA_WORKAROUND
	/*
	 * Indicate enable/disable workaround for CN31xx Pass 1.1 USBN's broken DMA.
	 */
	u_int8_t	sc_workaround_force_slave_mode; /** force to use slave mode */
#endif

	int sc_noport;
	u_int8_t sc_addr;		/* device address */
	u_int8_t sc_conf;		/* device configuration */
	struct usbd_xfer *sc_intrxfer;
	char sc_isreset;		/* USBC is reseted ? */
	char sc_connstat_change;	/* USBC is disconnected ? */
#define USBC_PRT_STAT_CONNECTED		0x01
#define USBC_PRT_STAT_DISCONNECTED	0x02
	char sc_isprtbbl;		/* USBC is port babble ? */
#ifdef USB_USE_SOFTINTR
	char sc_softwake;
#endif /* USB_USE_SOFTINTR */

	u_int16_t	sc_frame_number;
	u_int16_t	sc_frame_interval;

	octeon_usbc_soft_ed_t *sc_isoc_head;
	octeon_usbc_soft_ed_t *sc_ctrl_head;
	octeon_usbc_soft_ed_t *sc_intr_head;
	octeon_usbc_soft_ed_t *sc_bulk_head;

	octeon_usbc_soft_ed_t *sc_freeseds;
	octeon_usbc_soft_td_t *sc_freestds;

	SIMPLEQ_HEAD(, usbd_xfer) sc_free_xfers; /* free xfers */

	void                    *sc_ih;
#if defined(__NetBSD__)
	device_t sc_child;
#elif defined(__OpenBSD__)
	device_ptr_t sc_child;			/* /dev/usb# device */
#endif
	char sc_dying;

	u_int8_t sc_port_speed;
#ifndef USBC_HPRT_PRTSPD_HIGH
#define  USBC_HPRT_PRTSPD_HIGH		0x0
#endif
#ifndef USBC_HPRT_PRTSPD_FULL
#define  USBC_HPRT_PRTSPD_FULL		0x1
#endif
#ifndef USBC_HPRT_PRTSPD_LOW
#define  USBC_HPRT_PRTSPD_LOW			0x2
#endif
#ifndef USBC_HPRT_PRTSPD_RESERVED
#define  USBC_HPRT_PRTSPD_RESERVED		0x3
#endif


	u_int32_t sc_rx_fifo_size;
	u_int32_t sc_nperio_tx_fifo_siz;
	u_int32_t sc_perio_tx_fifo_siz;

	/* stored registors */
	u_int32_t sc_haint;
	u_int32_t sc_haintmsk;

	/* H/W Configuration */
	struct{
		u_int32_t ptxqdepth;
		u_int32_t nptxqdepth;
		u_int8_t dynfifosizing;
		u_int8_t periosupport;
		u_int8_t numhstch; /** number of host channels */

#if USBC_USE_DEV_MODE
		/* device mode params */
		u_int8_t num_dev_eps;
		u_int8_t num_dev_perio_eps;
#endif
	} sc_hwcfg;

	octeon_usbc_hostch_t *sc_hostch_list; /** host channel list */

	callout_t sc_tmo_hprt;
	callout_t sc_tmo_nptxfemp;

} octeon_usbc_softc_t;

struct octeon_usbc_pipe {
	struct usbd_pipe pipe;
	int nexttoggle;

	octeon_usbc_soft_ed_t *sed;

	union {
		octeon_usbc_soft_td_t *std;
	} tail;

	/* Info needed for different pipe kinds. */
	union {
		/* Control pipe */
		struct {
			usb_dma_t req_setup_dma;
			usb_dma_t req_stat_dma;
#define USBC_CONTROL_STATUS_BUF_SIZE 64
			u_int length;
			octeon_usbc_soft_td_t *setup, *data, *stat;
		} ctl;
		/* Interrupt pipe */
		struct {
			u_int length;
			int nslots;
			int pos;
		} intr;
		/* Bulk pipe */
		struct {
			u_int length;
			int isread;
			usb_dma_t term_dma;
#define USBC_BULK_TERMINATE_BUF_SIZE 64
		} bulk;
		/* Iso pipe */
		struct iso {
			int next, inuse;
		} iso;
	} u;
};


usbd_status octeon_usbc_init(struct octeon_usbc_softc *sc);

#endif /* _OCTEON_USBCVAR_H_ */