
Added initial support Xilinx Embedded Software. Signed-off-by: Jagannadha Sutradharudu Teki <jaganna@xilinx.com>
479 lines
16 KiB
C
Executable file
479 lines
16 KiB
C
Executable file
/******************************************************************************
|
|
*
|
|
* Copyright (C) 2006 Vreelin Engineering, Inc. All Rights Reserved.
|
|
* (c) Copyright 2007-2013 Xilinx, Inc. All rights reserved.
|
|
*
|
|
* This file contains confidential and proprietary information of Xilinx, Inc.
|
|
* and is protected under U.S. and international copyright and other
|
|
* intellectual property laws.
|
|
*
|
|
* DISCLAIMER
|
|
* This disclaimer is not a license and does not grant any rights to the
|
|
* materials distributed herewith. Except as otherwise provided in a valid
|
|
* license issued to you by Xilinx, and to the maximum extent permitted by
|
|
* applicable law: (1) THESE MATERIALS ARE MADE AVAILABLE "AS IS" AND WITH ALL
|
|
* FAULTS, AND XILINX HEREBY DISCLAIMS ALL WARRANTIES AND CONDITIONS, EXPRESS,
|
|
* IMPLIED, OR STATUTORY, INCLUDING BUT NOT LIMITED TO WARRANTIES OF
|
|
* MERCHANTABILITY, NON-INFRINGEMENT, OR FITNESS FOR ANY PARTICULAR PURPOSE;
|
|
* and (2) Xilinx shall not be liable (whether in contract or tort, including
|
|
* negligence, or under any other theory of liability) for any loss or damage
|
|
* of any kind or nature related to, arising under or in connection with these
|
|
* materials, including for any direct, or any indirect, special, incidental,
|
|
* or consequential loss or damage (including loss of data, profits, goodwill,
|
|
* or any type of loss or damage suffered as a result of any action brought by
|
|
* a third party) even if such damage or loss was reasonably foreseeable or
|
|
* Xilinx had been advised of the possibility of the same.
|
|
*
|
|
* CRITICAL APPLICATIONS
|
|
* Xilinx products are not designed or intended to be fail-safe, or for use in
|
|
* any application requiring fail-safe performance, such as life-support or
|
|
* safety devices or systems, Class III medical devices, nuclear facilities,
|
|
* applications related to the deployment of airbags, or any other applications
|
|
* that could lead to death, personal injury, or severe property or
|
|
* environmental damage (individually and collectively, "Critical
|
|
* Applications"). Customer assumes the sole risk and liability of any use of
|
|
* Xilinx products in Critical Applications, subject only to applicable laws
|
|
* and regulations governing limitations on product liability.
|
|
*
|
|
* THIS COPYRIGHT NOTICE AND DISCLAIMER MUST BE RETAINED AS PART OF THIS FILE
|
|
* AT ALL TIMES.
|
|
*
|
|
*******************************************************************************/
|
|
/******************************************************************************/
|
|
/**
|
|
* @file xusb_intr.c
|
|
*
|
|
* This file contains the functions that are related to interrupt processing
|
|
* for the USB device.
|
|
*
|
|
* <pre>
|
|
* MODIFICATION HISTORY:
|
|
*
|
|
* Ver Who Date Changes
|
|
* ----- ---- ------------------------------------------------------------------
|
|
* 1.00a hvm 2/22/07 First release.
|
|
* 2.00a hvm 12/2/08 Updated the interrupt handler for handling the
|
|
* new DMA bits defined in the interrupt status register.
|
|
* 3.00a hvm 12/3/09 Updated to use HAL processor APIs. Removed _m from the
|
|
* Added the interrrupt handling for the error interrupts.
|
|
* Added new API XUsb_ErrIntrSetHandler for setting up
|
|
* error handler.
|
|
* 3.01a hvm 5/20/10 Updated with fix for CR561171. The interrupt handler is
|
|
* updated to call the error handler callback function
|
|
* during error interrupts.
|
|
* 4.00a hvm 9/01/10 Added new API XUsb_DmaIntrSetHandler for setting up DMA
|
|
* handler. Updated the XUsb_IntrHandler function to call
|
|
* the DMA handler to handle DMA events. Removed the
|
|
* DmaDone and DmaError variable usage from the
|
|
* XUsb_IntrHandler function.
|
|
* Added interrupt handling for ULPI PHY interrupts.
|
|
* 4.02a bss 3/04/12 Modified XCOMPONENT_IS_READY to XIL_COMPONENT_IS_READY
|
|
* CR 650877
|
|
* </pre>
|
|
******************************************************************************/
|
|
|
|
/***************************** Include Files **********************************/
|
|
|
|
#include "xusb.h"
|
|
|
|
/************************** Constant Definitions ******************************/
|
|
|
|
/**************************** Type Definitions ********************************/
|
|
|
|
/***************** Macros (Inline Functions) Definitions **********************/
|
|
|
|
/************************** Variable Definitions ******************************/
|
|
|
|
/************************** Function Prototypes *******************************/
|
|
|
|
/****************************************************************************/
|
|
/**
|
|
*
|
|
* This function enables the specified interrupts.
|
|
*
|
|
* @param InstancePtr is a pointer to the XUsb instance.
|
|
* @param Mask is the bit-mask of the interrupts to be enabled.
|
|
* Bit positions of 1 will be enabled. Bit positions of 0 will
|
|
* keep the previous setting. This mask is formed by OR'ing
|
|
* XUSB_STATUS_*_MASK bits defined in xusb_l.h.
|
|
*
|
|
* @return None.
|
|
*
|
|
* @note None.
|
|
*
|
|
*****************************************************************************/
|
|
void XUsb_IntrEnable(XUsb *InstancePtr, u32 Mask)
|
|
{
|
|
u32 IntrValue;
|
|
|
|
Xil_AssertVoid(InstancePtr != NULL);
|
|
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
|
|
|
|
/*
|
|
* Write to the Interrupt Enable Register to enable the interrupts.
|
|
*/
|
|
IntrValue = XUsb_ReadReg(InstancePtr->Config.BaseAddress,
|
|
XUSB_IER_OFFSET);
|
|
IntrValue |= (Mask & (XUSB_STATUS_INTR_ALL_MASK));
|
|
XUsb_WriteReg(InstancePtr->Config.BaseAddress,
|
|
XUSB_IER_OFFSET, IntrValue);
|
|
|
|
}
|
|
|
|
/****************************************************************************/
|
|
/**
|
|
*
|
|
* This function disables the specified interrupts.
|
|
*
|
|
* @param InstancePtr is a pointer to the XUsb instance.
|
|
* @param Mask is the bit-mask of the interrupts to be disabled.
|
|
* Bit positions of 1 will be enabled. Bit positions of 0 will
|
|
* keep the previous setting. This mask is formed by OR'ing
|
|
* XUSB_STATUS_*_MASK bits defined in xusb_l.h.
|
|
*
|
|
* @return None.
|
|
*
|
|
* @note None.
|
|
*
|
|
*****************************************************************************/
|
|
void XUsb_IntrDisable(XUsb *InstancePtr, u32 Mask)
|
|
{
|
|
u32 IntrValue;
|
|
|
|
Xil_AssertVoid(InstancePtr != NULL);
|
|
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
|
|
|
|
/*
|
|
* Write to the Interrupt Enable Register to disable the interrupts.
|
|
*/
|
|
IntrValue = XUsb_ReadReg(InstancePtr->Config.BaseAddress,
|
|
XUSB_IER_OFFSET);
|
|
IntrValue &= (~(Mask & (XUSB_STATUS_INTR_ALL_MASK)));
|
|
XUsb_WriteReg(InstancePtr->Config.BaseAddress,
|
|
XUSB_IER_OFFSET, IntrValue);
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/**
|
|
*
|
|
* This function is the interrupt handler for the USB driver.
|
|
* This function is the first-level interrupt handler for the USB core. All USB
|
|
* interrupts will be handled here. Depending on the type of the interrupt,
|
|
* second level interrupt handler may be called. Second level interrupt
|
|
* handlers will be registered by the user using the XUsb_IntrSetHandler()
|
|
* and/or XUsb_EpSetHandler() functions.
|
|
*
|
|
* This handler reads the interrupt status from the Interrupt Status Register,
|
|
* determines the source of the interrupts, calls the corresponding callbacks and
|
|
* finally clears the interrupts.
|
|
*
|
|
* The interrupt from the USB driver has to be connected to the interrupt
|
|
* controller and the handler has to be registered with the interrupt system.
|
|
*
|
|
* Applications using this driver are responsible for providing the callbacks to
|
|
* handle interrupts and installing the callbacks using XUsb_SetHandler()
|
|
* during the initialization phase.
|
|
*
|
|
* @param InstancePtr is a pointer to the XUsb instance that just
|
|
* interrupted.
|
|
*
|
|
* @return None.
|
|
*
|
|
* @note No user handler is defined for the DMA interrupts.
|
|
* The DMA interrupt status is updated in the DMA status
|
|
* variables defined under the USB driver instance.
|
|
*
|
|
******************************************************************************/
|
|
void XUsb_IntrHandler(void *InstancePtr)
|
|
{
|
|
XUsb *UsbInstPtr;
|
|
u32 IntrStatus;
|
|
u32 IntrEnable;
|
|
u32 PendingIntr;
|
|
XUsb_EpConfig *Ep;
|
|
u8 Index;
|
|
|
|
Xil_AssertVoid(InstancePtr != NULL);
|
|
|
|
UsbInstPtr = (XUsb *) InstancePtr;
|
|
|
|
/*
|
|
* Read the Interrupt Enable Register.
|
|
*/
|
|
IntrEnable = XUsb_ReadReg(UsbInstPtr->Config.BaseAddress,
|
|
XUSB_IER_OFFSET);
|
|
|
|
/*
|
|
* Read the Interrupt Status Register.
|
|
*/
|
|
IntrStatus = XUsb_ReadReg(UsbInstPtr->Config.BaseAddress,
|
|
XUSB_STATUS_OFFSET);
|
|
|
|
|
|
if ((IntrStatus & XUSB_STATUS_HIGH_SPEED_MASK) ==
|
|
XUSB_STATUS_HIGH_SPEED_MASK) {
|
|
UsbInstPtr->DeviceConfig.CurrentSpeed =
|
|
XUSB_EP_HIGH_SPEED;
|
|
} else {
|
|
UsbInstPtr->DeviceConfig.CurrentSpeed =
|
|
XUSB_EP_FULL_SPEED;
|
|
}
|
|
|
|
PendingIntr = IntrStatus & IntrEnable;
|
|
|
|
/*
|
|
* Call the handler for the event interrupt.
|
|
*/
|
|
if (PendingIntr & XUSB_STATUS_INTR_EVENT_MASK) {
|
|
|
|
/*
|
|
* Check if there is any action to be done for :
|
|
* - USB Reset received {XUSB_STATUS_RESET_MASK}
|
|
* - USB Suspend received {XUSB_STATUS_SUSPEND_MASK}
|
|
* - USB Disconnect received {XUSB_STATUS_DISCONNECT_MASK}
|
|
* - USB SOF received {XUSB_STATUS_SOF_PACKET_MASK}
|
|
* - USB RESUME received {XUSB_STATUS_DMA_RESUME_MASK}
|
|
*/
|
|
if (IntrStatus & IntrEnable)
|
|
if (UsbInstPtr->HandlerFunc) {
|
|
UsbInstPtr->HandlerFunc(UsbInstPtr->HandlerRef,
|
|
IntrStatus);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Check the buffer completion interrupts .
|
|
*/
|
|
if (PendingIntr & XUSB_STATUS_INTR_BUFF_COMP_ALL_MASK) {
|
|
|
|
if (PendingIntr & XUSB_STATUS_EP0_BUFF1_COMP_MASK) {
|
|
|
|
Ep = &UsbInstPtr->DeviceConfig.Ep[XUSB_EP_NUMBER_ZERO];
|
|
if (Ep->HandlerFunc) {
|
|
Ep->HandlerFunc(Ep->HandlerRef,
|
|
XUSB_EP_NUMBER_ZERO, IntrStatus);
|
|
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Process the endpoint buffer interrupts.
|
|
*/
|
|
for (Index = 0; Index <
|
|
(UsbInstPtr->DeviceConfig.NumEndpoints - 1);
|
|
Index++) {
|
|
|
|
if ((PendingIntr &
|
|
(XUSB_STATUS_EP1_BUFF1_COMP_MASK << Index)) ||
|
|
(PendingIntr &
|
|
(XUSB_STATUS_EP1_BUFF2_COMP_MASK <<
|
|
Index))) {
|
|
|
|
Ep = &UsbInstPtr->DeviceConfig.Ep[Index + 1];
|
|
if (Ep->HandlerFunc) {
|
|
Ep->HandlerFunc(Ep->HandlerRef,
|
|
(Index + 1), (PendingIntr & (
|
|
(XUSB_STATUS_EP1_BUFF1_COMP_MASK |
|
|
XUSB_STATUS_EP1_BUFF2_COMP_MASK)
|
|
<< Index)));
|
|
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (UsbInstPtr->Config.DmaEnabled) {
|
|
|
|
if (PendingIntr & XUSB_STATUS_DMA_EVENT_MASK) {
|
|
|
|
/*
|
|
* Call the DMA event handler
|
|
*/
|
|
UsbInstPtr->DmaHandlerFunc(UsbInstPtr->DmaHandlerRef,
|
|
IntrStatus);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Call the error handler if any USB error has occured.
|
|
*
|
|
*/
|
|
if (PendingIntr & XUSB_STATUS_ERROR_EVENT_MASK) {
|
|
/*
|
|
* Call the error handler
|
|
*/
|
|
UsbInstPtr->ErrHandlerFunc(UsbInstPtr->ErrHandlerRef,
|
|
IntrStatus);
|
|
|
|
}
|
|
|
|
/*
|
|
* Call the ULPI PHY handler as the previous transaction is completed.
|
|
*
|
|
*/
|
|
if (PendingIntr & XUSB_STATUS_PHY_ACCESS_MASK) {
|
|
/*
|
|
* Call the ULPI PHY handler
|
|
*/
|
|
UsbInstPtr->UlpiHandlerFunc(UsbInstPtr->UlpiHandlerRef,
|
|
IntrStatus);
|
|
|
|
}
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/**
|
|
*
|
|
* This function installs an asynchronous callback function for the general
|
|
* interrupt (interrupts other than the endpoint and error interrupts).
|
|
*
|
|
* @param InstancePtr is a pointer to the XUsb instance.
|
|
* @param CallBackFunc is the address of the callback function.
|
|
* @param CallBackRef is a user data item that will be passed to the
|
|
* callback function when it is invoked.
|
|
*
|
|
* @return None.
|
|
*
|
|
* @note
|
|
* Invoking this function for a handler that already has been
|
|
* installed replaces it with the new handler. The user can disable
|
|
* a handler by setting the callback function pointer to NULL.
|
|
*
|
|
******************************************************************************/
|
|
void XUsb_IntrSetHandler(XUsb *InstancePtr, void *CallBackFunc,
|
|
void *CallBackRef)
|
|
{
|
|
Xil_AssertVoid(InstancePtr != NULL);
|
|
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
|
|
|
|
InstancePtr->HandlerFunc = (XUsb_IntrHandlerFunc) CallBackFunc;
|
|
InstancePtr->HandlerRef = CallBackRef;
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/**
|
|
* This function sets the handler for endpoint events.
|
|
*
|
|
* @param InstancePtr is a Pointer to the XUsb instance.
|
|
* @param EpNum is the Number of the endpoint.
|
|
* @param CallBackFunc is the address of the callback function.
|
|
* It can be NULL if the user wants to disable the handler entry.
|
|
* @param CallBackRef is a user data item that will be passed to the
|
|
* callback function when it is invoked. This can be NULL.
|
|
*
|
|
* @return None.
|
|
*
|
|
* @note
|
|
* Invoking this function for a handler that already has been
|
|
* installed replaces it with the new handler. The user can disable
|
|
* a handler by setting the callback function pointer to NULL.
|
|
*
|
|
******************************************************************************/
|
|
void XUsb_EpSetHandler(XUsb *InstancePtr, u8 EpNum,
|
|
XUsb_EpHandlerFunc *CallBackFunc, void *CallBackRef)
|
|
{
|
|
XUsb_EpConfig *Ep;
|
|
|
|
Xil_AssertVoid(InstancePtr != NULL);
|
|
Xil_AssertVoid(CallBackFunc != NULL);
|
|
Xil_AssertVoid(EpNum <= InstancePtr->DeviceConfig.NumEndpoints);
|
|
|
|
Ep = &InstancePtr->DeviceConfig.Ep[EpNum];
|
|
Ep->HandlerFunc = (XUsb_EpHandlerFunc) CallBackFunc;
|
|
Ep->HandlerRef = CallBackRef;
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/**
|
|
*
|
|
* This function installs an asynchronous callback function for the error
|
|
* events.
|
|
*
|
|
* @param InstancePtr is a pointer to the XUsb instance.
|
|
* @param CallBackFunc is the address of the callback function.
|
|
* @param CallBackRef is a user data item that will be passed to the
|
|
* callback function when it is invoked.
|
|
*
|
|
* @return None.
|
|
*
|
|
* @note
|
|
* Invoking this function for a handler that already has been
|
|
* installed replaces it with the new handler. The user can disable
|
|
* a handler by setting the callback function pointer to NULL.
|
|
*
|
|
******************************************************************************/
|
|
void XUsb_ErrIntrSetHandler(XUsb *InstancePtr, void *CallBackFunc,
|
|
void *CallBackRef)
|
|
{
|
|
Xil_AssertVoid(InstancePtr != NULL);
|
|
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
|
|
|
|
InstancePtr->ErrHandlerFunc = (XUsb_IntrHandlerFunc) CallBackFunc;
|
|
InstancePtr->ErrHandlerRef = CallBackRef;
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/**
|
|
*
|
|
* This function installs an asynchronous callback function for the DMA
|
|
* events.
|
|
*
|
|
* @param InstancePtr is a pointer to the XUsb instance.
|
|
* @param CallBackFunc is the address of the callback function.
|
|
* @param CallBackRef is a user data item that will be passed to the
|
|
* callback function when it is invoked.
|
|
*
|
|
* @return None.
|
|
*
|
|
* @note
|
|
* Invoking this function for a handler that already has been
|
|
* installed replaces it with the new handler. The user can disable
|
|
* a handler by setting the callback function pointer to NULL.
|
|
*
|
|
******************************************************************************/
|
|
void XUsb_DmaIntrSetHandler(XUsb *InstancePtr, void *CallBackFunc,
|
|
void *CallBackRef)
|
|
{
|
|
Xil_AssertVoid(InstancePtr != NULL);
|
|
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
|
|
|
|
InstancePtr->DmaHandlerFunc = (XUsb_IntrHandlerFunc) CallBackFunc;
|
|
InstancePtr->DmaHandlerRef = CallBackRef;
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/**
|
|
*
|
|
* This function installs an asynchronous callback function for the ULPI PHY
|
|
* events.
|
|
*
|
|
* @param InstancePtr is a pointer to the XUsb instance.
|
|
* @param CallBackFunc is the address of the callback function.
|
|
* @param CallBackRef is a user data item that will be passed to the
|
|
* callback function when it is invoked.
|
|
*
|
|
* @return None.
|
|
*
|
|
* @note
|
|
* Invoking this function for a handler that already has been
|
|
* installed replaces it with the new handler. The user can disable
|
|
* a handler by setting the callback function pointer to NULL.
|
|
*
|
|
******************************************************************************/
|
|
void XUsb_UlpiIntrSetHandler(XUsb *InstancePtr, void *CallBackFunc,
|
|
void *CallBackRef)
|
|
{
|
|
Xil_AssertVoid(InstancePtr != NULL);
|
|
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
|
|
|
|
InstancePtr->UlpiHandlerFunc = (XUsb_IntrHandlerFunc) CallBackFunc;
|
|
InstancePtr->UlpiHandlerRef = CallBackRef;
|
|
|
|
}
|