As long as this article is related to the introduction of TMS320C6000 and DSP, it focuses on the TMS320C6000 series interrupt setting problem in detail.
DSPWith the widespread application of DSP (digital signal processor) systems, the scale of their programs has also continued to expand. Using the Boot-loader of the chip itself to guide DSP programs through Flash memory is often restricted by the size and structure of the program, such as The program greatly exceeds the scope of the manufacturer's solidified boot, and the influence of the different positions of the interrupt vector table on the program boot jump, etc., so more and more flexible boot methods are needed.
After the system is powered on, the boot program will guide the DSP application from the memory to the high-speed memory (such as internal SRAM, SDRAM, etc.) on the DSP application board. Because the Flash memory has the function of deleting electrical signals, the deletion speed is fast, and the integration is high, it has become the first choice for this type of memory. Because the access speed of the Flash memory is slow, the program written into the Flash memory will be loaded into the fast memory by the DSP to run when the system is powered on. This process is called Boot loader. Different DSPs have different guidance methods.
DSP programmingThe boot loading of the DSP system refers to the process in which the system automatically transplants a section of code stored in the external non-volatile memory to the high-speed RAM of the internal DSP and executes it after the system is powered on.
Therefore, in the boot loading system, the performance of external non-volatile memory and DSP is particularly important.
FLASH memory is a high-density, non-volatile electrically erasable memory.
Moreover, the price of the unit storage bit is lower than that of the traditional EPROM, so it is very suitable as an external expansion memory.
Before the system is powered on, the boot program and user program must be written into FLASH.
When programming, in addition to using a dedicated hardware programmer to program FLASH, FLASH usually also supports DSP software programming to achieve the same function.
After the system is powered on, generally first run the boot program in FLASH, and complete the transplantation of the user program by itself, and then run the user program moved to the DSP chip at high speed by the DSP.
DSP programming skillsmposerStudio) is a complete DSP integrated development environment developed by TI. Because TI's DSP is widely used, CCS has become one of the most widely used DSP development software. Now, all TI DSPs can be developed in this environment, which can realize full-space transparent simulation without occupying any user resources. The software is equipped with assembly/linking, C compiler, C source code debugger, etc. Different systems have different versions of CCS development environment. The integrated development environment of TMS320F28l2 is CCS2000. When you buy a development board, you will give you CCS development software for free, and you can also download the software for free on TI’s website. The installation and use of the development environment are generally described in detail in the instruction manual of the DSP development board, so I won’t comment on it here. .
2. The programming language CCS development environment supports two languages: assembly language and C language. Each TI DSP series has a corresponding set of assembly instructions. If you use assembly language programming, you need to be familiar with these instructions. The difficulty and efficiency can be imagined. Now TI engineers are constantly improving CCS's C program optimization compiler, and the current C optimization efficiency can reach 90% or even higher than that of manual assembly. Of course, sometimes if computing power and memory resources are the bottleneck, ASM still has advantages, such as G.729 codec. But for general application development, C language is the best choice. As long as there is a certain C language foundation, DSP development and programming can be carried out.
Application examples and development process
1. Example introduction The main functions realized by the control part of the 400Hz inverter power supply are: (1) AD conversion, (2) SPWM generation, (3) serial communication with the display microcontroller, (4) CAN bus with other parallel power supplies Communication; (5) Capture synchronization signal. To properly realize the above functions, it is obvious that ordinary single-chip microcomputers cannot complete, so we chose the digital signal processor TMS320F2812 specially used for control. This instrument introduces the development process of TMS320F28l2 generating SPWM wave in detail.
PWM (PulseWidthModulation) control is a technique for modulating the width of a pulse, that is, by modulating the width of a series of pulses to equivalently obtain the desired waveform.
SPWM waveform (SinuSOIdalPWM) is a PWM waveform whose pulse width changes according to the sine law and is equivalent to a sine wave. It is the most widely used in inverter circuits. In this example, the function of SPWM is to provide trigger pulses to the 6 IGBTs (insulated gate bipolar transistors) of the three-phase inverter circuit, so as to invert the direct current into a 400Hz sinusoidal alternating current.
2. Development process (1) The generation of the writing software SPWM is mainly to use the event manager module (EVA/B) of TMS320F28l2, so the main work to be done before writing the program is to understand the structure and working principle of this module, as for other Modules that are not used may not be understood temporarily. The first step of programming is to set up various registers of the modules used, and the second step is to design and write algorithms.
(2) Debugging program After writing the program, debug it in the simulation environment, compile it first and then run it. You can directly observe the SPWM waveform output on the DSP pin with the oscilloscope until it matches the required waveform.
(3) Programming the program After debugging the program in the simulation environment, the program needs to be programmed into the on-chip FLASH memory of the DSP. When programming, you must first install the special programming software. After installation, you can directly operate the programming in the CCS environment. After the programming is completed, it can be separated from the emulator, and the development board can be powered on and run independently, and connected with other peripheral circuits to realize its functions.
Reference resources for learning methods 1. Beginners learning DSP often feel that there are too many technical documents to start with. According to my self-study experience, the most important thing to understand at this time is the working principle of the internal hardware of the DSP chip. If there is a microcomputer principle and a microcontroller foundation, this should be easy to grasp. If there is no foundation in this area, it is recommended to find it first. Read a book on the principles of microcomputers, and then look at the principles of DSP after you understand it. In addition, there are many DSP peripheral modules, so there is no need to understand them all at this time, just understand the modules you are using.
After understanding the working principle of DSP, it is necessary to read books about software development environment, including software installation and use. Then look for a few complete projects (the complete project of each module will be given away when you buy the development board). After you have a general understanding of the creation and structure of a complete project, you can start to analyze and use it. The module-related program. Now there are a large number of ready-made routines and algorithms for reference. Don't fumble yourself, be sure to find related programs for reference. This will achieve a multiplier effect with half the effort.
Talking about the Interrupt Setting of TMS320C6000 SeriesWhat general work needs to be done to realize DSP interrupt
Set which non-maskable interrupts are allowed
Set the interrupt source of each allowed non-masked interrupt
Set to turn on the total interrupt
Design interrupt vector table
Mount the interrupt vector table to the instruction memory through the cmd file
Provide interrupt handling function
If the first address of the interrupt vector table is not an address, then the interrupt vector table address register needs to be set
For different interrupt sources, you need to do your own work. For example, if it is an external interrupt, you need to set the pin polarity, that is, interrupt from high to low or vice versa.
In order to take care of readers with less knowledge, the following will start from a new project and guide everyone to create an interrupt sample program.
If you are familiar with building projects, you can skip this step.
3. Establish a new project
1. Click Project-》New, set Project Name to intexample, Project Type to Executable, Target select the device you need, because I am using DSK6416 evaluation board. So choose TMS320C64XX.
2. Add standard library rts6400.lib to automatically generate c_int00 and other functions. Right-click the current project, select "Add Files to Project", select the path where the library is located, usually the CCS installation comes with it, you can refer to the path address of this CCS3.1 version: \CCStudio_v3.1\C6000\cgtools\libts6400.lib
If you are using other device types, please select other device libraries in the lib folder.
Add a source file, select File-"New-"Source File, save it as main.c to the project path.
Write the main function in this file.
void main (void)
{
while(1);
}
Finally, add this file to the project through 2 steps.
3. Add the register alias definition header file. In this example, after defining aliases for the registers that need to be used, the global.h file is formed, and the content is introduced step by step in the following text. Here you can create an empty file and include it in main.c.
#I nclude “global.hâ€
At this point, a new engineering framework for DSP has been completed.
4. Add cmd link file
In order to realize the memory configuration at link time, we need to provide a cmd file. For convenience, you can copy a copy from the official sample program and modify it.
In the installation directory D:\CCStudio_v3.1utorial\device type\hello1 example, you will find a hello1.cmd,
Copy it to the project directory, rename it to link.cmd, and finally add it to the project.
Since this file does not declare stack and heap, a warning will be generated, and it will easily overflow if there are more dynamic data. Therefore, we'd better provide the size of stack and heap in this file. The value can be adjusted according to the actual situation. After modification, the content of this file is similar to:
-stack 0x1000
-heap 0x1000
MEMORY
{
ISRAM: origin = 0x0, len = 0x1000000
}
SECTIONS
{
.vectors》 ISRAM
.text》 ISRAM
.bss "ISRAM
.cinit》 ISRAM
.const》 ISRAM
.far》 ISRAM
.stack》 ISRAM
.cio》 ISRAM
.sysmem "ISRAM
}
At this point, the project is created, you can compile it again and observe whether it is normal.
--------------------------- intexample.pjt-Debug ------------------ ---------
[Main.c] "D:\CCStudio_v3.1\C6000\cgtools\bin\cl6x" -g -fr "D:/intexample/Debug" -d"_DEBUG" -mv6400 -@"Debug.lkf" "main. c"
[Linking.. . ] "D:\CCStudio_v3.1\C6000\cgtools\bin\cl6x" -@"Debug.lkf"
"Linking"
Build Complete,
0 Errors, 0 Warnings, 0 Remarks.
Fourth, the timer interrupt design
First of all, we first implement a timer interrupt, because it is not subject to external influence and is easy to test.
In the global.h file, add the control register and interrupt register alias definition. In addition, in order to use timer 1, the alias should also be defined:
/*Define the control register*/
extern cregister volatile unsigned int AMR; /* Address Mode Register */
extern cregister volatile unsigned int CSR; /* Control Status Register */
extern cregister volatile unsigned int IFR; /* Interrupt Flag Register */
extern cregister volatile unsigned int ISR; /* Interrupt Set Register */
extern cregister volatile unsigned int ICR; /* Interrupt Clear Register */
extern cregister volatile unsigned int IER; /* Interrupt Enable Register */
extern cregister volatile unsigned int ISTP; /* Interrupt Service Tbl Ptr */
extern cregister volatile unsigned int IRP; /* Interrupt Return Pointer */
extern cregister volatile unsigned int NRP; /* Non-maskable Int Return Ptr*/
extern cregister volatile unsigned int IN; /* General Purpose Input Reg */
extern cregister volatile unsigned int OUT; /* General Purpose Output Reg */
/* Define interrupt selection register*/
#define MUXH 0x019C0000
#define MUXL 0x019C0004
#define EXTPOL 0x019C0008
/*Define timer 1 register*/
#define CTL1 0x01980000 //Timer1 control register
#define PRD1 0x01980004 //Timer1 period register
#define CNT1 0x01980008 //Timer1 counter register
After that, initialize the timer in the main function. Here we use Timer1. The parameter initialization function is as follows:
void Timer1_Init(void)
{
*( volatile unsigned int*) CTL1= 0x00000201; //Counter function setting
*( volatile unsigned int*) PRD1= 0x1000; //Counter period value
*( volatile unsigned int*) CTL1|= 0x000000C0; //The counter is cleared and started
Comment on the previous sentence: * (volatile unsigned int*) CTL1=CTL1|0x000000C0
}
And call it in the main function.
Then we set the interrupt register parameters.
DSP supports 1 RESET interrupt, 1 NMI (non-maskable interrupt), 12 maskable interrupts (INT4-15), they have priority order, INT4 is the highest, INT15 is the lowest. Each interrupt number (INT4-INT15) can set any interrupt source. Here we select interrupt INT10, that is, turn on interrupt 10, and set its interrupt source to timer 1, that is, fill in the interrupt source selection code in the specified bit in MUXH or MUXL:
The interrupt source selection code is defined as follows: (This content can be obtained by searching for INTSEL in the help)
INTSEL (Interrupt Selection Number Deion)
00000b DSPINT Host port host to DSP interrupt
00001b TINT0 Timer 0 interrupt
00010b TINT1 Timer 1 interrupt
00011b SD_INT EMIF SDRAM timer interrupt
00100b EXT_INT4 External interrupt 4
00101b EXT_INT5 External interrupt 5
00110b EXT_INT6 External interrupt 6
00111b EXT_INT7 External interrupt 7
01000b EDMA_INT EDMA channel (0-15) interrupt
01001-01011b Reserved
01100b XINT0 McBSP0 transmit interrupt
01101b RINT0 McBSP0 receive interrupt
01110b XINT1 McBSP1 transmit interrupt
01111b RINT1 McBSP1 receive interrupt
10000-11111b Reserved
From this, the interrupt selection code for Timer 1 is 00010.
The register definitions of MUXH and MUXL are as follows: (also can be obtained through help)
MUXH
Bit interrupt source
30-26 INTSEL15
25-21 INTSEL14
20-16 INTSEL13
14-10 INTSEL12
9-5 INTSEL11
4-0 INTSEL10
31, 15 reserved, fill in
MUXL
Bit interrupt source
30-26 INTSEL9
25-21 INTSEL8
20-16 INTSEL7
14-10 INTSEL6
9-5 INTSEL5
4-0 INTSEL4 //Interrupt source of interrupt INT4, select timer interrupt
31, 15 reserved, fill in
Therefore, we set the 4-0 bits of MUXH as the interrupt selection code 00010 of timer 1, and the remaining bits can be set arbitrarily (1 can be filled here). After converting to hexadecimal, the settings are as follows:
*( volatile unsigned int*) MUXH=0x7fff7fe2;
MUXL does not need to be set.
Turn on the interrupt to IE10, and enable the global interrupt:
IER |= 0x0000 0402; // IE10=1
CSR |= 0x00000001; // Global interrupt enable
The above completes the configuration of the interrupt parameters, the interrupt is started and can be entered. The following is the interrupt processing process. It is mainly divided into design interrupt vector table and interrupt processing function.
We can copy a prototype of the vector table from the DSP CCS example. For example \CCStudio_v3.1utorial\dsk6416\hello1\vectors.asm
Copy it to the project directory and add it to the project.
The interrupt vector table contains 16 interrupt processing units, and each unit must be limited to 8 instructions. If there are not enough 8, you can fill it with nop (but nop 4 counts as 1 statement). If there are too many service routines, then you can make a special interrupt service routine. At this time, this table only serves as a jump function, so that the CPU can Correct addressing to find the correct interrupt service entry.
First analyze this file.
A macro is defined at the beginning of the file to handle unused interrupts.
unused .macro id
.global unused:id:
unused:id:
b unused:id:; nested branches to block interrupts
nop 4
b unused:id:
nop
nop
nop
nop
nop
Its approach is to let the program enter an endless loop, I think this approach is not necessarily optimal, so I recommend using the direct return method. Return to the interrupted address. For the maskable interrupt, it is birp, so replace this macro part with it. Pay attention to make up 8 entries.
unused .macro id
.global unused:id:
unused:id:
b irp
nop
nop
nop
nop
nop
nop
nop
.endm
In this way, even if we turn on this interrupt by mistake, we will return smoothly. Of course, if we are sure that it is indeed not turned on, then its content is meaningless.
The body of the code uses a series of unused n to insert this macro, which plays a role in occupying space.
Since the return of NMI is different from the maskable interrupt, it is located under RESET in the vector table, that is, unused 1. We delete it and replace it with
NMI: b irp
nop
nop
nop
nop
nop
nop
nop
In order to realize the processing of timer 1 interrupt, we delete unused 10 and replace it with our own interrupt jump program, as follows:
INT10:
stw b0, *--b15
mvkl _xint0_isr, b0
mvkh _xint0_isr, b0
b b0
ldw *b15++, b0
nop 3
nop
nop
In addition, the need and statement:
.ref _c_int00; C entry point
Similarly, add a reference to the handler
.ref _xint0_isr; timer 1 interrupt handler
Since the location of the interrupt vector table needs to be specified specifically, and should be aligned to 400H, in this file, the segment name has been defined:
.sect ".vectors"
Therefore, we need to mount this .vector code segment to a special designated memory area.
Modify the link.cmd link file and add the INT area, starting from the address. Its size is 400H, modify the original ISRAM starting point. And point the .vector in SECTIONS to the memory area defined by yourself.
MEMORY
{
INT: origin = 0x00000000, len = 0x0000400
ISRAM: origin = 0x00000400, len = 0x1000000
}
SECTIONS
{
.vectors "INT
…
}
The interrupt vector table is set and installed.
Finally, design the interrupt service function and add in main.c:
interrupt void xint0_isr(void)
{
}
Note that the interrupt keyword must be identified to generate the interrupt return statement b irp. At the same time, the entry and exit parameters of this function should be void. If you need to update variables, you can use global variables.
In addition, the C language function name differs from the assembly by a "_", please pay attention to adding it when designing the interrupt vector table.
After the above steps, the entire timer interrupt production process is completed. At this point, you can add a breakpoint on interrupt void xint0_isr (void), and it should stop here after running. If the entry fails, you can first set a breakpoint on the INT10:stw b0, *--b15 sentence of vector.asm. If you do not enter here, it proves that the interrupt has not entered. You can check whether there is a problem with the parameter setting.
Five, external interrupt design
The DSP6000 series provides INT4-7 four interrupt input pins, so external interrupts can be realized through the input level changes of these four pins. The polarity of the level change is divided into high to low and low to high. Therefore, the DSP uses the register EXTPOL to set. Only the lower 4 bits of EXTPOL are valid, representing INT4-7 respectively. For each bit, there are:
: Low-"High interrupt
1: High-"Low generates an interrupt
So set it to complete the polarity change.
Below, take the setting of the external port INT7 interrupt and mount it to the 12th interrupt as an example to briefly describe the implementation process:
Set the 12th interrupt to external interrupt 7, that is, MUXH (14:10)=00111, at this time MUXH is set to:
*( volatile unsigned int*) MUXH=0x7fff7ce2;//0111 1100 1110 0010
Turn on the 12th bit of IER.
IER |= 0x00001402; // IE10=1 IE12=1
Replace unused 12 of vectors.asm with:
INT12:
stw b0, *--b15
mvkl _extint7_isr, b0
mvkh _extint7_isr, b0
b b0
ldw *b15++, b0
nop 3
nop
nop
And add a reference
.ref _extint7_isr
Add the service function in main.c:
interrupt void extint7_isr(void)
{
}
In hardware, if a low->high signal is generated on the INT7/GPIO7 pin, an interrupt can be triggered.
If you change this polarity, you can set the fourth bit of EXTPOL to 1:
*( volatile unsigned int*) EXTPOL|= 0x00000008;
At this time, a high->low signal can generate an interrupt.
It should be noted that if you have initialized the GPIO, you must ensure that the corresponding bit of the GPEN interrupt pin is 1. If all are enabled:
*(Volatile unsigned int* )GPEN = 0x000000F0;
Six, MCBSP serial port receiving interrupt design
In the actual application process, it is often necessary to receive serial port data through interrupts. It is assumed that the MCBSP0 receive interrupt is added to the 11th.
First, add the MCBSP0 alias to the global.h file.
Set the MCBSP0 parameter and enable it, and its initialization function is:
void MCBSP0_Init(void)
{
*( volatile unsigned int*) McBSP0_SPCR = 0x00000000;
*( volatile unsigned int*) McBSP0_SRGR = 0x200000FF;
*( volatile unsigned int*) McBSP0_PCR = 0x00000800;
*( volatile unsigned int*) McBSP0_XCR = 0x000100A0;
*( volatile unsigned int*) McBSP0_RCR = 0x000100A0;
*( volatile unsigned int*) McBSP0_MCR = 0x00000000;
*( volatile unsigned int*) McBSP0_SPCR |= 0x00C10001;
}
And call it in the main function.
Turn on interrupt 11:
IER |= 0x00001C02; // IE10=1 IE11=1 IE12=1
And set MUXH(9:5)=01101, combining the above three interrupts,
At this time, MUXH is: *(volatile unsigned int*) MUXH=0x7fff1da2;//0001 1101 1010 0010
Of course, if you only consider the current interrupt, MUXH can be set to:
*( volatile unsigned int*) MUXH=0x7fff7dbf;//0111 1101 1011 1111
Make an interrupt service program and take out the data:
interrupt void rint0_isr(void)
{
int DotRev;
DotRev=*( volatile unsigned int *) McBSP0_DRR;
}
Modify vectors.asm and replace unused 11 with:
INT11:
stw b0, *--b15
mvkl _rint0_isr, b0
mvkh _rint0_isr, b0
b b0
ldw *b15++, b0
nop 3
nop
nop
Add reference:
.ref _rint0_isr; mcbsp 0 receive interrupt handler
At this time, all tasks are completed, you can observe the received value by setting a breakpoint.
Also need to pay attention to the data must be taken out in the service program, otherwise it will stop receiving new data.
Seven, other topics
1. Set the start position of the interrupt vector table
The above discussion is to place the interrupt vector table at the address. If it needs to be placed at any address (aligned with 400H), then the starting address of the vector table needs to be provided.
For example, our terminal vector position: INT is set to:
MEMORY
{
INT: origin = 0x00000400, len = 0x0000400
ISRAM: origin = 0x00000800, len = 0x1000000
}
Then when we initialize the interrupt, we should set:
ISTP=0x00000400;
2. View the current interrupt bitmap
You can check the corresponding bits (15:0) of the interrupt flag register IFR to see if an interrupt arrives.
3. Clear/set the original interrupt
If you need to clear the original interrupt, you can set the corresponding bit in the ICR register. If you want to manually trigger an interrupt, you can set the corresponding bit in the ISR register, and they will update the IFR bitmap.
For example, in the timer interrupt service program, we manually trigger the external INT7 interrupt No. 12 by setting the 12th bit of the ISR.
interrupt void xint0_isr(void)
{
ISR=0x00001000;
}
Then the CPU will execute extint7_isr (void) to handle this interrupt.
For another example, in the external interrupt in the above example, sometimes there is a situation that the interrupt will come in when the signal is not sent as soon as the power is turned on, so how to overcome it? It can be overcome by ICR. Set ICR to clear the maskable interrupt. The corresponding bit is valid. For example, when the interrupt initialization is set, all the original interrupts are cleared. Then you can add a statement:
ICR =0xffffffff;
4. Interrupt settings under DSP/BIOS
Under the DSP/BIOS management, we don't need to set the interrupt vector table and interrupt initialization by ourselves. Everything can be done through the graphical settings of the BIOS.
Add a DSP/BIOS
Choose File-》New, choose DSK6416 in this test, readers can choose according to their actual needs. Save it as Configuration1.cdb.
Add it to the project.
As required in the above example, select HWI interrupts No. 10, 11, and 12, right-click and select Properties to fill in the following parameters:
HWI_INT10 interrupt source=Timer_1 =_xint0_isr (note the underscore!)
HWI_INT11 interrupt source=MCSP_0_Receive =_rint0_isr
HWI_INT11 interrupt source=External_Pin_7 =_extint7_isr
The interrupt can be started in the same way in the main function.
IER |= 0x00001C02; // IE10=1 IE11=1 IE12=1
CSR |= 0x00000001; //Global interrupt enable
The configuration is now complete.
5. How to check if the interruption can't come in?
First check whether the corresponding bit of IER is set to open, and the lowest bit of CSR is set. Secondly, check whether the interrupt vector table address is set correctly. If the confirmation is correct. Set a breakpoint where the vector table interrupt should enter. Run to see if the execution reaches the breakpoint. If so, then see if the interrupt service routine is executed. If the interrupt can no longer be entered after it comes in only once, you can check whether the interrupt vector table can return to the original program. If it cannot return, check whether there are 8 statements. In addition, you can check whether the birp statement is executed by tracing. If you can return to the original program normally, such as serial port reception, see if there is no value that causes blocking. If this is the case, the original value needs to be taken out to have a new interrupt.
6. Where can I find the description of several interrupt registers?
You can get it through Help-》Contents, search keywords. You can also refer to official documents.
ConclusionThis is the end of the related introduction about TMS320C6000. Please correct me if there are any deficiencies.
Related reading recommendations: tms320c6000 series dsp programming tools and guides
Related reading recommendations: Optimized design of Viterbi decoding program based on TMS320C6000 series DSP
Ultra Transparency High Resolution Sky Curtain
Ultra Transparency High Resolution Sky Curtain,Led Ad Wall Display Screen,Outdoor Led Digital Signage,Application Of Display Grille Screen
Kindwin Technology (H.K.) Limited , https://www.ktlleds.com