Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Dependents: Bootcamp-dma-serial
Fork of MODDMA by
Diff: MODDMA.cpp
- Revision:
- 8:cb4d323ce6fd
- Parent:
- 6:40d38be4bb59
- Child:
- 12:1dfee7208043
--- a/MODDMA.cpp Wed Nov 24 00:32:49 2010 +0000
+++ b/MODDMA.cpp Sat Feb 05 08:53:26 2011 +0000
@@ -1,150 +1,157 @@
-/*
- Copyright (c) 2010 Andy Kirkham
-
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
-
- The above copyright notice and this permission notice shall be included in
- all copies or substantial portions of the Software.
-
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- THE SOFTWARE.
-*/
-
-#include "MODDMA.h"
-
-namespace AjK {
-
-// Create a "hook" for our ISR to make callbacks. Set by init()
-class MODDMA *moddma_p = (class MODDMA *)NULL;
-
-void
-MODDMA::Enable(CHANNELS ChannelNumber)
-{
- LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
- pChannel->DMACCConfig |= _E;
-}
-
-bool
-MODDMA::Enabled(CHANNELS ChannelNumber)
-{
- LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
- return (bool)(pChannel->DMACCConfig & _E);
-}
-
-void
-MODDMA::Disable(CHANNELS ChannelNumber)
-{
- LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
- pChannel->DMACCConfig &= ~(_E);
-}
-
-bool
-MODDMA::isActive(CHANNELS ChannelNumber)
-{
- LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
- return (bool)( pChannel->DMACCConfig & CxConfig_A() ) ;
-}
-
-void
-MODDMA::haltChannel(CHANNELS ChannelNumber)
-{
- LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
- pChannel->DMACCConfig |= CxConfig_H();
-}
-
-uint32_t oldDMAHandler = 0;
-typedef void (*MODDMA_FN)(void);
-
-extern "C" void MODDMA_IRQHandler(void) {
- uint32_t channel_mask;
-
- if (moddma_p == (class MODDMA *)NULL) {
- if (oldDMAHandler) {
- ((MODDMA_FN)oldDMAHandler)();
- return;
- }
- else {
- error("Interrupt without instance");
- }
- }
-
- for (int channel_number = 0; channel_number < 8; channel_number++) {
- channel_mask = (1UL << channel_number);
- if (LPC_GPDMA->DMACIntStat & channel_mask) {
- if (LPC_GPDMA->DMACIntTCStat & channel_mask) {
- if (moddma_p->setups[channel_number] != (MODDMA_Config *)NULL) {
- moddma_p->setIrqProcessingChannel((MODDMA::CHANNELS)channel_number);
- moddma_p->setIrqType(MODDMA::TcIrq);
- moddma_p->setups[channel_number]->isrIntTCStat->call();
- moddma_p->isrIntTCStat.call();
- // The user callback should clear the IRQ. But if they forget
- // then the Mbed will lockup. So, check to see if the IRQ has
- // been dismissed, if not, we will dismiss it here.
- if (LPC_GPDMA->DMACIntTCStat & channel_mask) {
- LPC_GPDMA->DMACIntTCClear = channel_mask;
- }
- // If the user has left the channel enabled, disable it.
- // Not, we don't check Active here as it may block inside
- // an ISR, we just shut it down immediately. If the user
- // must wait for completion they should implement their
- // own ISR. But only disable if the LLI linked list register
- // is null otherwise we can crap out a series of transfers.
- if (moddma_p->Enabled( (MODDMA::CHANNELS)channel_number )) {
- if (moddma_p->lli(channel_number) == 0 ) {
- moddma_p->Disable( (MODDMA::CHANNELS)channel_number );
- }
- }
- }
- }
-
- if (LPC_GPDMA->DMACIntErrStat & channel_mask) {
- if (moddma_p->setups[channel_number] != (MODDMA_Config *)NULL) {
- moddma_p->setIrqProcessingChannel((MODDMA::CHANNELS)channel_number);
- moddma_p->setIrqType(MODDMA::ErrIrq);
- moddma_p->setups[channel_number]->isrIntErrStat->call();
- moddma_p->isrIntErrStat.call();
- // The user callback should clear the IRQ. But if they forget
- // then the Mbed will lockup. So, check to see if the IRQ has
- // been dismissed, if not, we will dismiss it here.
- if (LPC_GPDMA->DMACIntErrStat & channel_mask) {
- LPC_GPDMA->DMACIntErrClr = channel_mask;
- }
- // If the user has left the channel enabled, disable it.
- // Not, we don't check Active here as it may block inside
- // an ISR, we just shut it down immediately. If the user
- // must wait for completion they should implement their
- // own ISR. But only disable if the LLI linked list register
- // is null otherwise we can crap out a series of transfers.
- if (moddma_p->Enabled( (MODDMA::CHANNELS)channel_number )) {
- if (moddma_p->lli(channel_number) == 0 ) {
- moddma_p->Disable( (MODDMA::CHANNELS)channel_number );
- }
- }
- }
- }
- }
- }
-
- /* IRQ should be handled by now, check to make sure. */
- if (LPC_GPDMA->DMACIntStat) {
- ((MODDMA_FN)oldDMAHandler)();
- LPC_GPDMA->DMACIntTCClear = (uint32_t)0xFF; /* If not, clear anyway! */
- }
- if (LPC_GPDMA->DMACIntErrStat) {
- ((MODDMA_FN)oldDMAHandler)();
- LPC_GPDMA->DMACIntErrClr = (uint32_t)0xFF; /* If not, clear anyway! */
- }
-}
-
-}; // namespace AjK ends
-
+/*
+ Copyright (c) 2010 Andy Kirkham
+
+ Permission is hereby granted, free of charge, to any person obtaining a copy
+ of this software and associated documentation files (the "Software"), to deal
+ in the Software without restriction, including without limitation the rights
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ copies of the Software, and to permit persons to whom the Software is
+ furnished to do so, subject to the following conditions:
+
+ The above copyright notice and this permission notice shall be included in
+ all copies or substantial portions of the Software.
+
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ THE SOFTWARE.
+*/
+#include "iomacros.h"
+#include "MODDMA.h"
+
+namespace AjK {
+
+// Create a "hook" for our ISR to make callbacks. Set by init()
+class MODDMA *moddma_p = (class MODDMA *)NULL;
+
+void
+MODDMA::Enable(CHANNELS ChannelNumber)
+{
+ LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
+ pChannel->DMACCConfig |= _E;
+}
+
+bool
+MODDMA::Enabled(CHANNELS ChannelNumber)
+{
+ LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
+ return (bool)(pChannel->DMACCConfig & _E);
+}
+
+void
+MODDMA::Disable(CHANNELS ChannelNumber)
+{
+ LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
+ pChannel->DMACCConfig &= ~(_E);
+}
+
+bool
+MODDMA::isActive(CHANNELS ChannelNumber)
+{
+ LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
+ return (bool)( pChannel->DMACCConfig & CxConfig_A() ) ;
+}
+
+void
+MODDMA::haltChannel(CHANNELS ChannelNumber)
+{
+ LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
+ pChannel->DMACCConfig |= CxConfig_H();
+}
+
+uint32_t
+MODDMA::getControl(CHANNELS ChannelNumber)
+{
+ LPC_GPDMACH_TypeDef *pChannel = (LPC_GPDMACH_TypeDef *)Channel_p( ChannelNumber );
+ return pChannel->DMACCControl;
+}
+
+uint32_t oldDMAHandler = 0;
+typedef void (*MODDMA_FN)(void);
+
+extern "C" void MODDMA_IRQHandler(void) {
+ uint32_t channel_mask;
+
+ if (moddma_p == (class MODDMA *)NULL) {
+ if (oldDMAHandler) {
+ ((MODDMA_FN)oldDMAHandler)();
+ return;
+ }
+ else {
+ error("Interrupt without instance");
+ }
+ }
+
+ for (int channel_number = 0; channel_number < 8; channel_number++) {
+ channel_mask = (1UL << channel_number);
+ if (LPC_GPDMA->DMACIntStat & channel_mask) {
+ if (LPC_GPDMA->DMACIntTCStat & channel_mask) {
+ if (moddma_p->setups[channel_number] != (MODDMA_Config *)NULL) {
+ moddma_p->setIrqProcessingChannel((MODDMA::CHANNELS)channel_number);
+ moddma_p->setIrqType(MODDMA::TcIrq);
+ moddma_p->setups[channel_number]->isrIntTCStat->call();
+ moddma_p->isrIntTCStat.call();
+ // The user callback should clear the IRQ. But if they forget
+ // then the Mbed will lockup. So, check to see if the IRQ has
+ // been dismissed, if not, we will dismiss it here.
+ if (LPC_GPDMA->DMACIntTCStat & channel_mask) {
+ LPC_GPDMA->DMACIntTCClear = channel_mask;
+ }
+ // If the user has left the channel enabled, disable it.
+ // Note, we don't check Active here as it may block inside
+ // an ISR, we just shut it down immediately. If the user
+ // must wait for completion they should implement their
+ // own ISR. But only disable if the LLI linked list register
+ // is null otherwise we can crap out a series of transfers.
+ if (moddma_p->Enabled( (MODDMA::CHANNELS)channel_number )) {
+ if (moddma_p->lli( (MODDMA::CHANNELS)channel_number ) == 0 ) {
+ moddma_p->Disable( (MODDMA::CHANNELS)channel_number );
+ }
+ }
+ }
+ }
+
+ if (LPC_GPDMA->DMACIntErrStat & channel_mask) {
+ if (moddma_p->setups[channel_number] != (MODDMA_Config *)NULL) {
+ moddma_p->setIrqProcessingChannel((MODDMA::CHANNELS)channel_number);
+ moddma_p->setIrqType(MODDMA::ErrIrq);
+ moddma_p->setups[channel_number]->isrIntErrStat->call();
+ moddma_p->isrIntErrStat.call();
+ // The user callback should clear the IRQ. But if they forget
+ // then the Mbed will lockup. So, check to see if the IRQ has
+ // been dismissed, if not, we will dismiss it here.
+ if (LPC_GPDMA->DMACIntErrStat & channel_mask) {
+ LPC_GPDMA->DMACIntErrClr = channel_mask;
+ }
+ // If the user has left the channel enabled, disable it.
+ // Not, we don't check Active here as it may block inside
+ // an ISR, we just shut it down immediately. If the user
+ // must wait for completion they should implement their
+ // own ISR. But only disable if the LLI linked list register
+ // is null otherwise we can crap out a series of transfers.
+ if (moddma_p->Enabled( (MODDMA::CHANNELS)channel_number )) {
+ if (moddma_p->lli( (MODDMA::CHANNELS)channel_number ) == 0 ) {
+ moddma_p->Disable( (MODDMA::CHANNELS)channel_number );
+ }
+ }
+ }
+ }
+ }
+ }
+
+ /* IRQ should be handled by now, check to make sure. */
+ if (LPC_GPDMA->DMACIntStat) {
+ ((MODDMA_FN)oldDMAHandler)();
+ LPC_GPDMA->DMACIntTCClear = (uint32_t)0xFF; /* If not, clear anyway! */
+ }
+ if (LPC_GPDMA->DMACIntErrStat) {
+ ((MODDMA_FN)oldDMAHandler)();
+ LPC_GPDMA->DMACIntErrClr = (uint32_t)0xFF; /* If not, clear anyway! */
+ }
+}
+
+}; // namespace AjK ends
+
