129 lines
3.3 KiB
C
129 lines
3.3 KiB
C
|
//
|
||
|
// synchronize.c
|
||
|
//
|
||
|
// USPi - An USB driver for Raspberry Pi written in C
|
||
|
// Copyright (C) 2014-2015 R. Stange <rsta2@o2online.de>
|
||
|
//
|
||
|
// This program is free software: you can redistribute it and/or modify
|
||
|
// it under the terms of the GNU General Public License as published by
|
||
|
// the Free Software Foundation, either version 3 of the License, or
|
||
|
// (at your option) any later version.
|
||
|
//
|
||
|
// This program is distributed in the hope that it will be useful,
|
||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
// GNU General Public License for more details.
|
||
|
//
|
||
|
// You should have received a copy of the GNU General Public License
|
||
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
//
|
||
|
#include <uspi/synchronize.h>
|
||
|
#include <uspi/types.h>
|
||
|
#include <uspi/assert.h>
|
||
|
|
||
|
#define EnableInterrupts() __asm volatile ("cpsie i")
|
||
|
#define DisableInterrupts() __asm volatile ("cpsid i")
|
||
|
|
||
|
static volatile unsigned s_nCriticalLevel = 0;
|
||
|
static volatile boolean s_bWereEnabled;
|
||
|
|
||
|
void uspi_EnterCritical (void)
|
||
|
{
|
||
|
u32 nFlags;
|
||
|
asm volatile ("mrs %0, cpsr" : "=r" (nFlags));
|
||
|
|
||
|
DisableInterrupts ();
|
||
|
|
||
|
if (s_nCriticalLevel++ == 0)
|
||
|
{
|
||
|
s_bWereEnabled = nFlags & 0x80 ? FALSE : TRUE;
|
||
|
}
|
||
|
|
||
|
DataMemBarrier ();
|
||
|
}
|
||
|
|
||
|
void uspi_LeaveCritical (void)
|
||
|
{
|
||
|
DataMemBarrier ();
|
||
|
|
||
|
assert (s_nCriticalLevel > 0);
|
||
|
if (--s_nCriticalLevel == 0)
|
||
|
{
|
||
|
if (s_bWereEnabled)
|
||
|
{
|
||
|
EnableInterrupts ();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#if RASPPI == 1
|
||
|
|
||
|
//
|
||
|
// Cache maintenance operations for ARMv6
|
||
|
//
|
||
|
// NOTE: The following functions should hold all variables in CPU registers. Currently this will be
|
||
|
// ensured using maximum optimation (see circle/synchronize.h).
|
||
|
//
|
||
|
// The following numbers can be determined (dynamically) using CTR.
|
||
|
// As long we use the ARM1176JZF-S implementation in the BCM2835 these static values will work:
|
||
|
//
|
||
|
|
||
|
#define DATA_CACHE_LINE_LENGTH 32
|
||
|
|
||
|
void uspi_CleanAndInvalidateDataCacheRange (u32 nAddress, u32 nLength)
|
||
|
{
|
||
|
nLength += DATA_CACHE_LINE_LENGTH;
|
||
|
|
||
|
while (1)
|
||
|
{
|
||
|
asm volatile ("mcr p15, 0, %0, c7, c14, 1" : : "r" (nAddress) : "memory");
|
||
|
|
||
|
if (nLength < DATA_CACHE_LINE_LENGTH)
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
nAddress += DATA_CACHE_LINE_LENGTH;
|
||
|
nLength -= DATA_CACHE_LINE_LENGTH;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#else
|
||
|
|
||
|
//
|
||
|
// Cache maintenance operations for ARMv7-A
|
||
|
//
|
||
|
// See: ARMv7-A Architecture Reference Manual, Section B4.2.1
|
||
|
//
|
||
|
// NOTE: The following functions should hold all variables in CPU registers. Currently this will be
|
||
|
// ensured using the register keyword and maximum optimation (see uspi/synchronize.h).
|
||
|
//
|
||
|
// The following numbers can be determined (dynamically) using CTR, CSSELR, CCSIDR and CLIDR.
|
||
|
// As long we use the Cortex-A7 implementation in the BCM2836 or the Cortex-A53 implementation
|
||
|
// in the BCM2837 these static values will work:
|
||
|
//
|
||
|
|
||
|
#define L1_DATA_CACHE_LINE_LENGTH 64
|
||
|
#define L2_CACHE_LINE_LENGTH 64
|
||
|
#define DATA_CACHE_LINE_LENGTH_MIN 64 // min(L1_DATA_CACHE_LINE_LENGTH, L2_CACHE_LINE_LENGTH)
|
||
|
|
||
|
void uspi_CleanAndInvalidateDataCacheRange (u32 nAddress, u32 nLength)
|
||
|
{
|
||
|
nLength += DATA_CACHE_LINE_LENGTH_MIN;
|
||
|
|
||
|
while (1)
|
||
|
{
|
||
|
__asm volatile ("mcr p15, 0, %0, c7, c14, 1" : : "r" (nAddress) : "memory"); // DCCIMVAC
|
||
|
|
||
|
if (nLength < DATA_CACHE_LINE_LENGTH_MIN)
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
nAddress += DATA_CACHE_LINE_LENGTH_MIN;
|
||
|
nLength -= DATA_CACHE_LINE_LENGTH_MIN;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#endif
|