/*
 * This file is part of the libpayload project.
 *
 * Copyright (C) 2010 Patrick Georgi
 *
 * 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. The name of the author may not be used to endorse or promote products
 *    derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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 __XHCI_PRIVATE_H
#define __XHCI_PRIVATE_H

#include <usb/usb.h>

#define MASK(startbit, lenbit) (((1<<(lenbit))-1)<<(startbit))

typedef volatile union trb {
	// transfer

	// events
#define TRB_EV_CMD_CMPL 33
	struct {
		u32 Cmd_TRB_Pointer_lo;
		u32 Cmd_TRB_Pointer_hi;
		struct {
			unsigned long:24;
			unsigned long Completion_Code:8;
		} __attribute__ ((packed));
		struct {
			unsigned long C:1;
			unsigned long:9;
			unsigned long TRB_Type:6;
			unsigned long VF_ID:8;
			unsigned long Slot_ID:8;
		} __attribute__ ((packed));
	} __attribute__ ((packed)) event_cmd_cmpl;

#define TRB_EV_PORTSC 34
	struct {
		struct {
			unsigned long:24;
			unsigned long Port:8;
		} __attribute__ ((packed));
		u32 rsvd;
		struct {
			unsigned long:24;
			unsigned long Completion_Code:8;
		} __attribute__ ((packed));
		struct {
			unsigned long C:1;
			unsigned long:9;
			unsigned long TRB_Type:6;
			unsigned long:16;
		} __attribute__ ((packed));
	} __attribute__ ((packed)) event_portsc;

	// commands
#define TRB_CMD_NOOP 23
	struct {
		u32 rsvd[3];
		struct {
			unsigned long C:1;
			unsigned long:9;
			unsigned long TRB_Type:6;
			unsigned long:16;
		} __attribute__ ((packed));
	} __attribute__ ((packed)) cmd_No_Op;

	// "others"
	struct {
		u32 Ring_Segment_Ptr_lo;
		u32 Ring_Segment_Ptr_hi;
		struct {
			unsigned long:22;
			unsigned long Interrupter_Target;
		} __attribute__ ((packed));
		struct {
			unsigned long C:1;
			unsigned long TC:1;
			unsigned long:2;
			unsigned long CH:1;
			unsigned long IOC:1;
			unsigned long:4;
			unsigned long TRB_Type:6;
			unsigned long:16;
		} __attribute__ ((packed));
	} __attribute__ ((packed)) link;
} trb_t;

typedef struct slotctx {
	struct {
		unsigned long Route_String:20;
		unsigned long Speed:4;
		unsigned long:1;
		unsigned long MTT:1;
		unsigned long Hub:1;
		unsigned long Context_Entries:5;
	} __attribute__ ((packed));
	struct {
		unsigned long Max_Exit_Latency:16;
		unsigned long Root_Hub_Port_Number:8;
		unsigned long Number_of_Ports:8;
	} __attribute__ ((packed));
	struct {
		unsigned long TT_Hub_Slot_ID:8;
		unsigned long TT_Port_Number:8;
		unsigned long TTT:2;
		unsigned long:4;
		unsigned long Interrupter_Target:10;
	} __attribute__ ((packed));
	struct {
		unsigned long USB_Device_Address:8;
		unsigned long:19;
		unsigned long Slot_State:5;
	} __attribute__ ((packed));
	u32 rsvd[4];
} slotctx_t;

typedef struct epctx {
	struct {
		unsigned long EP_State:3;
		unsigned long:5;
		unsigned long Mult:2;
		unsigned long MaxPStreams:5;
		unsigned long LSA:1;
		unsigned long Interval:8;
		unsigned long:8;
	} __attribute__ ((packed));
	struct {
		unsigned long:1;
		unsigned long CErr:2;
		unsigned long EP_Type:3;
		unsigned long:1;
		unsigned long HID:1;
		unsigned long Max_Burst_Size:8;
		unsigned long Max_Packet_Size:16;
	} __attribute__ ((packed));
	union {
		u32 TR_Dequeue_Pointer_lo;
		struct {
			unsigned long DCS:1;
			unsigned long:3;
		} __attribute__ ((packed));
	} __attribute__ ((packed));
	u32 TR_Dequeue_Pointer_hi;
	struct {
		unsigned long Average_TRB_Length:16;
		unsigned long Max_ESIT_Payload:16;
	} __attribute__ ((packed));
	u32 rsvd[3];
} epctx_t;

typedef struct devctx {
	slotctx_t slot;
	epctx_t ep0;
	struct {
		epctx_t out;
		epctx_t in;
	} eps[15];
} devctx_t;

typedef struct devctxp {
	devctx_t *ptr;
	void *upper;
} devctxp_t;

typedef struct erst_entry {
	u32 seg_base_lo;
	u32 seg_base_hi;
	u32 seg_size;
	u32 rsvd;
} erst_entry_t;

typedef struct xhci {
	/* capreg is read-only, so no need for volatile,
	   and thus 32bit accesses can be assumed. */
	struct capreg {
		u8 caplength;
		u8 res1;
		union {
			u16 hciversion;
			struct {
				u8 hciver_lo;
				u8 hciver_hi;
			} __attribute__ ((packed));
		} __attribute__ ((packed));
		union {
			u32 hcsparams1;
			struct {
				unsigned long MaxSlots:7;
				unsigned long MaxIntrs:11;
				unsigned long:6;
				unsigned long MaxPorts:8;
			} __attribute__ ((packed));
		} __attribute__ ((packed));
		union {
			u32 hcsparams2;
			struct {
				unsigned long IST:4;
				unsigned long ERST_Max:4;
				unsigned long:18;
				unsigned long SPR:1;
				unsigned long Max_Scratchpad_Bufs:5;
			} __attribute__ ((packed));
		} __attribute__ ((packed));
		union {
			u32 hcsparams3;
			struct {
				unsigned long u1latency:8;
				unsigned long:8;
				unsigned long u2latency:16;
			} __attribute__ ((packed));
		} __attribute__ ((packed));
		union {
			u32 hccparams;
			struct {
				unsigned long ac64:1;
				unsigned long bnc:1;
				unsigned long csz:1;
				unsigned long ppc:1;
				unsigned long pind:1;
				unsigned long lhrc:1;
				unsigned long ltc:1;
				unsigned long nss:1;
				unsigned long:4;
				unsigned long MaxPSASize:4;
				unsigned long xECP:16;
			} __attribute__ ((packed));
		} __attribute__ ((packed));
		u32 dboff;
		u32 rtsoff;
	} __attribute__ ((packed)) *capreg;

	/* opreg is R/W is most places, so volatile access is necessary.
	   volatile means that the compiler seeks byte writes if possible,
	   making bitfields unusable for MMIO register blocks. Yay C :-( */
	volatile struct opreg {
		u32 usbcmd;
#define USBCMD_RS 1<<0
#define USBCMD_HCRST 1<<1
		u32 usbsts;
#define USBSTS_HCH 1<<0
#define USBSTS_HSE 1<<2
#define USBSTS_EINT 1<<3
#define USBSTS_PCD 1<<4
#define USBSTS_CNR 1<<11
		u32 pagesize;
		u8 res1[0x13-0x0c+1];
		u32 dnctrl;
		u32 crcr_lo;
		u32 crcr_hi;
#define CRCR_RCS 1<<0
#define CRCR_CS 1<<1
#define CRCR_CA 1<<2
#define CRCR_CRR 1<<3
		u8 res2[0x2f-0x20+1];
		u32 dcbaap_lo;
		u32 dcbaap_hi;
		u32 config;
#define CONFIG_MASK_MaxSlotsEn 0xff
		u8 res3[0x3ff-0x3c+1];
		struct {
			u32 portsc;
#define PORTSC_CCS 1<<0
#define PORTSC_PED 1<<1
	// BIT 2 rsvdZ
#define PORTSC_OCA 1<<3
#define PORTSC_PR 1<<4
#define PORTSC_PLS 1<<5
#define PORTSC_PLS_MASK MASK(5, 4)
#define PORTSC_PP 1<<9
#define PORTSC_PORT_SPEED 1<<10
#define PORTSC_PORT_SPEED_MASK MASK(10, 4)
#define PORTSC_PIC 1<<14
#define PORTSC_PIC_MASK MASK(14, 2)
#define PORTSC_LWS 1<<16
#define PORTSC_CSC 1<<17
#define PORTSC_PEC 1<<18
#define PORTSC_WRC 1<<19
#define PORTSC_OCC 1<<20
#define PORTSC_PRC 1<<21
#define PORTSC_PLC 1<<22
#define PORTSC_CEC 1<<23
#define PORTSC_CAS 1<<24
#define PORTSC_WCE 1<<25
#define PORTSC_WDE 1<<26
#define PORTSC_WOE 1<<27
	// BIT 29:28 rsvdZ
#define PORTSC_DR 1<<30
#define PORTSC_WPR 1<<31
#define PORTSC_RW_MASK PORTSC_PR | PORTSC_PLS_MASK | PORTSC_PP | PORTSC_PIC_MASK | PORTSC_LWS | PORTSC_WCE | PORTSC_WDE | PORTSC_WOE
			u32 portpmsc;
			u32 portli;
			u32 res;
		} __attribute__ ((packed)) prs[];
	} __attribute__ ((packed)) *opreg;

	/* R/W, volatile, MMIO -> no bitfields */
	volatile struct hcrreg {
		u32 mfindex;
		u8 res1[0x20-0x4];
		struct {
			u32 iman;
			u32 imod;
			u32 erstsz;
			u32 res;
			u32 erstba_lo;
			u32 erstba_hi;
			u32 erdp_lo;
			u32 erdp_hi;
		} __attribute__ ((packed)) intrrs[]; // up to 1024, but maximum host specific, given in capreg->MaxIntrs
	} __attribute__ ((packed)) *hcrreg;

	/* R/W, volatile, MMIO -> no bitfields */
	volatile u32 *dbreg;

	/* R/W, volatile, Memory -> bitfields allowed */
	volatile devctxp_t *dcbaa;

	trb_t *cmd_ring;
	trb_t *ev_ring;
	volatile erst_entry_t *ev_ring_table;
	int cmd_ccs, ev_ccs;

	usbdev_t *roothub;
} xhci_t;

#define XHCI_INST(controller) ((xhci_t*)((controller)->instance))

#endif