This patch enables the OHCI driver to use DMA memory, which is necessary for ARM systems where DMA devices are not cache coherent. I really only need this to test some later OHCI changes, but it was easy enough... copied almost verbatim from ehci.c. Change-Id: Ia717eef28340bd6182a6782e83bfdd0693cf0db1 Signed-off-by: Julius Werner <jwerner@chromium.org> Reviewed-on: https://chromium-review.googlesource.com/193730 Reviewed-by: Stefan Reinauer <reinauer@chromium.org> (cherry picked from commit e46b6ebc439e86a00e13bf656d60cf6c186a3777) Signed-off-by: Isaac Christensen <isaac.christensen@se-eng.com> Reviewed-on: http://review.coreboot.org/7010 Tested-by: build bot (Jenkins) Reviewed-by: Patrick Georgi <pgeorgi@google.com>
		
			
				
	
	
		
			267 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			267 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * 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 __OHCI_PRIVATE_H
 | 
						|
#define __OHCI_PRIVATE_H
 | 
						|
 | 
						|
#include <usb/usb.h>
 | 
						|
 | 
						|
#define MASK(startbit, lenbit) (((1<<(lenbit))-1)<<(startbit))
 | 
						|
 | 
						|
	// FIXME: fake
 | 
						|
	typedef enum { CMD} reg;
 | 
						|
 | 
						|
	enum {
 | 
						|
		NumberDownstreamPorts = 1<<0,
 | 
						|
		PowerSwitchingMode = 1<<8,
 | 
						|
		NoPowerSwitching = 1<<9,
 | 
						|
		DeviceType = 1<<10,
 | 
						|
		OverCurrentProtectionMode = 1<<11,
 | 
						|
		NoOverCurrentProtection = 1<<12,
 | 
						|
		PowerOnToPowerGoodTime = 1<<24
 | 
						|
	} HcRhDescriptorAReg;
 | 
						|
 | 
						|
	enum {
 | 
						|
		NumberDownstreamPortsMask = MASK(0, 8),
 | 
						|
		PowerOnToPowerGoodTimeMask = MASK(24, 8)
 | 
						|
	} HcRhDescriptorAMask;
 | 
						|
 | 
						|
	enum {
 | 
						|
		DeviceRemovable = 1<<0,
 | 
						|
		PortPowerControlMask = 1<<16
 | 
						|
	} HcRhDescriptorBReg;
 | 
						|
 | 
						|
	enum {
 | 
						|
		CurrentConnectStatus		= 1<<0,
 | 
						|
		PortEnableStatus		= 1<<1,
 | 
						|
		PortSuspendStatus		= 1<<2,
 | 
						|
		PortOverCurrentIndicator	= 1<<3,
 | 
						|
		PortResetStatus			= 1<<4,
 | 
						|
		PortPowerStatus			= 1<<8,
 | 
						|
		LowSpeedDeviceAttached		= 1<<9,
 | 
						|
		ConnectStatusChange		= 1<<16,
 | 
						|
		PortEnableStatusChange		= 1<<17,
 | 
						|
		PortSuspendStatusChange		= 1<<18,
 | 
						|
		PortOverCurrentIndicatorChange	= 1<<19,
 | 
						|
		PortResetStatusChange		= 1<<20
 | 
						|
	} HcRhPortStatusRead;
 | 
						|
	enum {
 | 
						|
		ClearPortEnable			= 1<<0,
 | 
						|
		SetPortEnable			= 1<<1,
 | 
						|
		SetPortSuspend			= 1<<2,
 | 
						|
		ClearSuspendStatus		= 1<<3,
 | 
						|
		SetPortReset			= 1<<4,
 | 
						|
		SetPortPower			= 1<<8,
 | 
						|
		ClearPortPower			= 1<<9,
 | 
						|
	} HcRhPortStatusSet;
 | 
						|
 | 
						|
	enum {
 | 
						|
		LocalPowerStatus = 1<<0,
 | 
						|
		OverCurrentIndicator = 1<<1,
 | 
						|
		DeviceRemoteWakeupEnable = 1<<15,
 | 
						|
		LocalPowerStatusChange = 1<<16,
 | 
						|
		OverCurrentIndicatorChange = 1<<17,
 | 
						|
		ClearRemoteWakeupEnable = 1<<31
 | 
						|
	} HcRhStatusReg;
 | 
						|
 | 
						|
	enum {
 | 
						|
		FrameInterval = 1<<0,
 | 
						|
		FSLargestDataPacket = 1<<16,
 | 
						|
		FrameIntervalToggle = 1<<31
 | 
						|
	} HcFmIntervalOffset;
 | 
						|
	enum {
 | 
						|
		FrameIntervalMask = MASK(0, 14),
 | 
						|
		FSLargestDataPacketMask = MASK(16, 15),
 | 
						|
		FrameIntervalToggleMask = MASK(31, 1)
 | 
						|
	} HcFmIntervalMask;
 | 
						|
 | 
						|
	enum {
 | 
						|
		ControlBulkServiceRatio = 1<<0,
 | 
						|
		PeriodicListEnable = 1<<2,
 | 
						|
		IsochronousEnable = 1<<3,
 | 
						|
		ControlListEnable = 1<<4,
 | 
						|
		BulkListEnable = 1<<5,
 | 
						|
		HostControllerFunctionalState = 1<<6,
 | 
						|
		InterruptRouting = 1<<8,
 | 
						|
		RemoteWakeupConnected = 1<<9,
 | 
						|
		RemoteWakeupEnable = 1<<10
 | 
						|
	} HcControlReg;
 | 
						|
 | 
						|
	enum {
 | 
						|
		ControlBulkServiceRatioMask = MASK(0, 2),
 | 
						|
		HostControllerFunctionalStateMask = MASK(6, 2)
 | 
						|
	} HcControlMask;
 | 
						|
 | 
						|
	enum {
 | 
						|
		USBReset = 0*HostControllerFunctionalState,
 | 
						|
		USBResume = 1*HostControllerFunctionalState,
 | 
						|
		USBOperational = 2*HostControllerFunctionalState,
 | 
						|
		USBSuspend = 3*HostControllerFunctionalState
 | 
						|
	};
 | 
						|
 | 
						|
	enum {
 | 
						|
		HostControllerReset = 1<<0,
 | 
						|
		ControlListFilled = 1<<1,
 | 
						|
		BulkListFilled = 1<<2,
 | 
						|
		OwnershipChangeRequest = 1<<3,
 | 
						|
		SchedulingOverrunCount = 1<<16
 | 
						|
	} HcCommandStatusReg;
 | 
						|
 | 
						|
	enum {
 | 
						|
		SchedulingOverrunCountMask = MASK(16, 2)
 | 
						|
	} HcCommandStatusMask;
 | 
						|
 | 
						|
	enum {
 | 
						|
		FrameRemaining = 1<<0,
 | 
						|
		FrameRemainingToggle = 1<<31
 | 
						|
	} HcFmRemainingReg;
 | 
						|
 | 
						|
	enum {
 | 
						|
		SchedulingOverrung = 1<<0,
 | 
						|
		WritebackDoneHead = 1<<1,
 | 
						|
		StartofFrame = 1<<2,
 | 
						|
		ResumeDetected = 1<<3,
 | 
						|
		UnrecoverableError = 1<<4,
 | 
						|
		FrameNumberOverflow = 1<<5,
 | 
						|
		RootHubStatusChange = 1<<6,
 | 
						|
		OwnershipChange = 1<<30
 | 
						|
	} HcInterruptStatusReg;
 | 
						|
 | 
						|
     typedef struct {
 | 
						|
	// Control and Status Partition
 | 
						|
	volatile u32 HcRevision;
 | 
						|
	volatile u32 HcControl;
 | 
						|
	volatile u32 HcCommandStatus;
 | 
						|
	volatile u32 HcInterruptStatus;
 | 
						|
	volatile u32 HcInterruptEnable;
 | 
						|
	volatile u32 HcInterruptDisable;
 | 
						|
 | 
						|
	// Memory Pointer Partition
 | 
						|
	volatile u32 HcHCCA;
 | 
						|
	volatile u32 HcPeriodCurrentED;
 | 
						|
	volatile u32 HcControlHeadED;
 | 
						|
	volatile u32 HcControlCurrentED;
 | 
						|
	volatile u32 HcBulkHeadED;
 | 
						|
	volatile u32 HcBulkCurrentED;
 | 
						|
	volatile u32 HcDoneHead;
 | 
						|
 | 
						|
	// Frame Counter Partition
 | 
						|
	volatile u32 HcFmInterval;
 | 
						|
	volatile u32 HcFmRemaining;
 | 
						|
	volatile u32 HcFmNumber;
 | 
						|
	volatile u32 HcPeriodicStart;
 | 
						|
	volatile u32 HcLSThreshold;
 | 
						|
 | 
						|
	// Root Hub Partition
 | 
						|
	volatile u32 HcRhDescriptorA;
 | 
						|
	volatile u32 HcRhDescriptorB;
 | 
						|
	volatile u32 HcRhStatus;
 | 
						|
	/* all bits in HcRhPortStatus registers are R/WC, so
 | 
						|
	   _DO NOT_ use |= to set the bits,
 | 
						|
	   this clears the entire state */
 | 
						|
	volatile u32 HcRhPortStatus[];
 | 
						|
     } __attribute__ ((packed)) opreg_t;
 | 
						|
 | 
						|
	typedef struct { /* should be 256 bytes according to spec */
 | 
						|
		u32 HccaInterruptTable[32];
 | 
						|
		volatile u16 HccaFrameNumber;
 | 
						|
		volatile u16 HccaPad1;
 | 
						|
		volatile u32 HccaDoneHead;
 | 
						|
		u8 reserved[116]; /* pad according to spec */
 | 
						|
		u8 what[4]; /* really pad to 256 as spec only covers 252 */
 | 
						|
	} __attribute__ ((packed)) hcca_t;
 | 
						|
 | 
						|
	typedef volatile struct {
 | 
						|
		u32 config;
 | 
						|
		u32 tail_pointer;
 | 
						|
		u32 head_pointer;
 | 
						|
		u32 next_ed;
 | 
						|
	} __attribute__ ((packed)) ed_t;
 | 
						|
#define ED_HALTED 1
 | 
						|
#define ED_TOGGLE 2
 | 
						|
 | 
						|
#define ED_FUNC_SHIFT 0
 | 
						|
#define ED_FUNC_MASK MASK(0, 7)
 | 
						|
#define ED_EP_SHIFT 7
 | 
						|
#define ED_EP_MASK MASK(7, 4)
 | 
						|
#define ED_DIR_SHIFT 11
 | 
						|
#define ED_DIR_MASK MASK(11, 2)
 | 
						|
#define ED_LOWSPEED (1 << 13)
 | 
						|
#define ED_MPS_SHIFT 16
 | 
						|
 | 
						|
	typedef volatile struct {
 | 
						|
		u32 config;
 | 
						|
		u32 current_buffer_pointer;
 | 
						|
		u32 next_td;
 | 
						|
		u32 buffer_end;
 | 
						|
	} __attribute__ ((packed)) td_t;
 | 
						|
/*
 | 
						|
 * Bits 0 through 17 of .config won't be interpreted by the host controller
 | 
						|
 * (HC) and, after processing the TD, the HC has to ensure those bits have
 | 
						|
 * the same state as before. So we are free to use those bits for our own
 | 
						|
 * purpose.
 | 
						|
 */
 | 
						|
#define TD_QUEUETYPE_SHIFT	0
 | 
						|
#define TD_QUEUETYPE_MASK	MASK(TD_QUEUETYPE_SHIFT, 2)
 | 
						|
#define TD_QUEUETYPE_ASYNC	(0 << TD_QUEUETYPE_SHIFT)
 | 
						|
#define TD_QUEUETYPE_INTR	(1 << TD_QUEUETYPE_SHIFT)
 | 
						|
 | 
						|
#define TD_DIRECTION_SHIFT 19
 | 
						|
#define TD_DIRECTION_MASK MASK(TD_DIRECTION_SHIFT, 2)
 | 
						|
#define TD_DIRECTION_SETUP OHCI_SETUP << TD_DIRECTION_SHIFT
 | 
						|
#define TD_DIRECTION_IN OHCI_IN << TD_DIRECTION_SHIFT
 | 
						|
#define TD_DIRECTION_OUT OHCI_OUT << TD_DIRECTION_SHIFT
 | 
						|
#define TD_DELAY_INTERRUPT_SHIFT	21
 | 
						|
#define TD_DELAY_INTERRUPT_MASK		MASK(TD_DELAY_INTERRUPT_SHIFT, 3)
 | 
						|
#define TD_DELAY_INTERRUPT_ZERO		0
 | 
						|
#define TD_DELAY_INTERRUPT_NOINTR	(7 << TD_DELAY_INTERRUPT_SHIFT)
 | 
						|
#define TD_TOGGLE_DATA0 0
 | 
						|
#define TD_TOGGLE_DATA1 (1 << 24)
 | 
						|
#define TD_TOGGLE_FROM_ED 0
 | 
						|
#define TD_TOGGLE_FROM_TD (1 << 25)
 | 
						|
#define TD_CC_SHIFT 28
 | 
						|
#define TD_CC_MASK MASK(TD_CC_SHIFT, 4)
 | 
						|
#define TD_CC_NOERR 0
 | 
						|
#define TD_CC_NOACCESS (14 << TD_CC_SHIFT) /* the lower of the two values, so "no access" can be tested with >= */
 | 
						|
 | 
						|
#define OHCI_INST(controller) ((ohci_t*)((controller)->instance))
 | 
						|
 | 
						|
	typedef struct ohci {
 | 
						|
		opreg_t *opreg;
 | 
						|
		hcca_t *hcca;
 | 
						|
		usbdev_t *roothub;
 | 
						|
		ed_t *periodic_ed;
 | 
						|
#define DMA_SIZE (64 * 1024)
 | 
						|
		void *dma_buffer;
 | 
						|
	} ohci_t;
 | 
						|
 | 
						|
	typedef enum { OHCI_SETUP=0, OHCI_OUT=1, OHCI_IN=2, OHCI_FROM_TD=3 } ohci_pid_t;
 | 
						|
 | 
						|
#endif
 |