VGA pixel grouping on STM32 - graphics
I have some code that displays a single pixel on screen through VGA but am a bit stuck on how I could set multiple pixels on screen where I want them. I set up two Timers for Vertical Sync and Horizontal Sync then using the V-Sync interrupt, I set a flag to allow PA8 to toggle and output a pixel at the correct timing based on the SetCompare value I set on the timer's channel. The STM32f103c8 is also overclocked to 128MHz. Here's the code:
#include "Arduino.h"
//640x480 at 60Hz
static volatile int vflag = 0;
void setup() {
#define FLASH_ACR (*(volatile uint32_t*)(0x40022000))
FLASH_ACR = 0b110010; //enable flash prefetch and wait state to increase stability at higher freq
pinMode(PA0, PWM); //31,468.75Hz (Horizontal Sync) (Channel 1)
Timer2.pause();
Timer2.setOverflow(4067); //reload register value
Timer2.setPrescaleFactor(1); //number that divides main clock
Timer2.setCompare(1, 488); //12% duty cycle (Syncpulse/Wholeline)
Timer2.setCompare(2, 2000); //0-4067 = vertical line going left or right respectively
Timer2.attachInterrupt(2, TRIGGER);
Timer2.refresh();
Timer2.resume();
pinMode(PA6, PWM); //60Hz (Vertical Sync) (Channel 1)
Timer3.pause();
Timer3.setOverflow(4183); //reload register value
Timer3.setPrescaleFactor(510); //number that divides main clock
Timer3.setCompare(1, 16); //0.38% duty cycle (Syncpulse/Wholeframe)
Timer3.setCompare(2, 2000); //0-4183 = horizontal line going up or down respectively
Timer3.attachInterrupt(2, TRIGGER2);
Timer3.refresh();
Timer3.resume();
pinMode(PA8, OUTPUT); //need to set PinMode in order for the ODR register to work
}
void loop() {
}
void TRIGGER(){
if(vflag==1){
__asm__ volatile (
"ldr r0, =(0x4001080C) \n\t" //GPIOA base address is 0x40010800 and ODR offset is 0x0C
"ldr r1, =(1<<8) \n\t" //turn on PA8
"ldr r2, =0 \n\t" //turn off PA8
"str r1, [r0] \n\t" //turn on PA8
"str r2, [r0] \n\t" //turn off PA8
);
vflag = 0; //we set the vflag back to zero when were done outputing pixels.
}
}
I understand there's graphical defects/glitches and the code can be improved on but I'm trying to focus on how in theory this works. What I want to do is have a word display on screen, that word will be made up of letters, and those letters will be made up of groups of pixels. So then whats the best (or simplest) way to group pixels and execute them multiple times on-screen? Or how is this usually done?
I do not code for STM32 so even the code looks foreign to me however it sounds like you are hard-coding the individual pixels with timer... and generating VGA signal by some GPIO. That combination of methods is problematic to use for programmable graphics.
I am using AVR32 (UC3A with much slower clock then yours) to doing VGA image using:
screen buffer (videoram)
simply I have entire screen image stored in MCU memory. So you can simply change it contents without changing the VGA output code ...
The problem is you need to have enough memory for the image (encoded in a way to enable direct transfer to VGA connector). I am using AVR32 with 16+32+32 KByte of RAM but most MCUs have much less RAM (static images can be stored in EPROM but then it would not be possible to change the image output). So in case you do not have enough either lower resolution to fit into memory or add external memory to your system.
use integrated HW peripherial for VGA signal generation
I have more versions of VGA signal generation working:
SDRAM ... using SDRAM interface of the MCU
SMC ... using SMC interface of the MCU
SSC ... using synchronous serial interface
It all boils down to copying the screen buffer to IO interface at the VGA dot clock (~30MHz). Then on HW side combining the used MCU pins into VGA signals
for speed you can use DMA.
On top of all this you need to generate the sync signals and that is all.
So look at your MCU datasheet and find interface capable of synchronous transfer at least 3 bit data (R,G,B) with VGA dot clock speed. The closer the clock is to VGA dot clock the better (as some VGA monitors and LCDs do not tolerate too big difference)
the sync signals can be hardcoded or even encoded in the video ram.
The fastest is to use serial interface but the output is just B&W instead of RGB (unless you got 3 SSC units/channels) however you can send entire 8/16/32 pixels at once or by DMA directly so the MCU has time for other stuff and also requires much less VRAM.
My favourite is SDRAM interface (using just its data bus)
Here image from mine system:
Mine interconnection looks like this (using MCUs SDRAM interface version):
VGA <- AT32UC3A0512
R PX10 (EBI_D0)
G PX09 (EBI_D1)
B PX08 (EBI_D2)
Bright PX07 (EBI_D3)*
HS PA03
VS PA04
Here the relevant source:
VGA_EBI_SDRAMC.h:
//------------------------------------------------------------------------------------------------
#define _PA_VGA_HS 8
#define _PA_VGA_VS 16
#define _PAmo 24
volatile avr32_gpio_port_t *port_PA=&GPIO.port[AVR32_PIN_PA00>>5];
volatile U8 *SDRAM=(U8*)AVR32_EBI_CS0_ADDRESS;
//------------------------------------------------------------------------------------------------
//--- VGA 640x480x4 60Hz -------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
#define VRAM_xs 304
#define VRAM_ys 400
#define VRAM_bs 4
#define VRAM_ls (VRAM_xs>>1)
U8 VRAM[VRAM_ls*VRAM_ys];
U8 VRAM_empty[VRAM_ls];
// Horizontal timing [us]
#define VGA_t0 3
#define VGA_t1 5
#define VGA_t2 32
// Vertikal timing [lines ~31.817us] aby voslo viac bodov tak je to natiahnute na 32us++
#define VGA_ys 525
#define VGA_VS 2
#define VGA_y0 (36+40)
#define VGA_y1 (VGA_y0+VRAM_ys)
//------------------------------------------------------------------------------------------------
void VGA_init();
void VGA_screen();
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
void VGA_init()
{
system_init();
Disable_global_interrupt();
gpio_configure_pins(port_PA,_PAmo,GPIO_DIR_OUTPUT|GPIO_INIT_HIGH);
static const gpio_map_t EBI_GPIO_MAP[] =
{
{ AVR32_EBI_DATA_0_PIN, AVR32_EBI_DATA_0_FUNCTION},
{ AVR32_EBI_DATA_1_PIN, AVR32_EBI_DATA_1_FUNCTION},
{ AVR32_EBI_DATA_2_PIN, AVR32_EBI_DATA_2_FUNCTION},
{ AVR32_EBI_DATA_3_PIN, AVR32_EBI_DATA_3_FUNCTION},
};
gpio_enable_module(EBI_GPIO_MAP, sizeof(EBI_GPIO_MAP) / sizeof(EBI_GPIO_MAP[0]));
AVR32_SDRAMC.mr=0; // normal mode
AVR32_SDRAMC.tr=0; // no refresh (T=0)
AVR32_SDRAMC.cr=
(AVR32_SDRAMC_CR_NC_11_COLUMN_BITS <<AVR32_SDRAMC_CR_NC_OFFSET)
|(AVR32_SDRAMC_CR_NR_13_ROW_BITS <<AVR32_SDRAMC_CR_NR_OFFSET)
|(AVR32_SDRAMC_CR_NB_TWO_BANKS <<AVR32_SDRAMC_CR_NB_OFFSET)
|(AVR32_SDRAMC_CAS_ONE_CYCLE <<AVR32_SDRAMC_CR_CAS_OFFSET)
|(AVR32_SDRAMC_DBW_16_BITS <<AVR32_SDRAMC_CR_DBW_OFFSET)
|(0 <<AVR32_SDRAMC_TWR_OFFSET)
|(0 <<AVR32_SDRAMC_TRC_OFFSET)
|(0 <<AVR32_SDRAMC_TRP_OFFSET)
|(0 <<AVR32_SDRAMC_TRCD_OFFSET)
|(0 <<AVR32_SDRAMC_TRAS_OFFSET)
|(0 <<AVR32_SDRAMC_TXSR_OFFSET);
AVR32_SDRAMC.hsr=AVR32_SDRAMC_HSR_DA_MASK;
AVR32_SDRAMC.mdr=AVR32_SDRAMC_MDR_MD_SDRAM;
// map SDRAM CS -> memory space
AVR32_HMATRIX.sfr[AVR32_EBI_HMATRIX_NR]|=1<<AVR32_EBI_SDRAM_CS;
AVR32_HMATRIX.sfr[AVR32_EBI_HMATRIX_NR];
U32 a;
for (a=0;a<VRAM_ls*VRAM_ys;a++) VRAM[a]=0;
for (a=0;a<VRAM_ls;a++) VRAM_empty[a]=0;
}
//------------------------------------------------------------------------------------------------
void VGA_screen()
{
U32 a,x,y,c,PA,t0;
wait_start(t0);
for (;;)
{
for (PA=_PAmo,a=0,y=0;y<VGA_ys;y++)
{
wait_start(t0);
if (y== 0) PA^=_PA_VGA_VS; else PA^=0; // VS on
if (y==VGA_VS) PA^=_PA_VGA_VS; else PA^=0; // VS off
PA^=_PA_VGA_HS; // HS on
port_PA->ovrc=PA^_PAmo;
port_PA->ovrs=PA;
wait_us(t0,VGA_t0);
PA^=_PA_VGA_HS; // HS off
port_PA->ovrc=PA^_PAmo;
port_PA->ovrs=PA;
wait_us(t0,VGA_t1);
*SDRAM=0; // blank (black)
if ((y>=VGA_y0)&&(y<VGA_y1))
for (x=0;x<VRAM_ls;x++)
{
c=VRAM[a];
*SDRAM=c>>4; // write pixel into SDRAM interface (address is ignored as I use only data bus pins)
a++;
*SDRAM=c; // write pixel into SDRAM interface (address is ignored as I use only data bus pins)
}
*SDRAM=0; // blank (black)
wait_us(t0,VGA_t2);
}
}
}
//------------------------------------------------------------------------------------------------
Main.cpp:
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
#include "System\include.h"
#include "pic_zilog_inside.h"
//#include "VGA_EBI_SMC.h"
#include "VGA_EBI_SDRAMC.h"
//#include "VGA_SSC.h"
//------------------------------------------------------------------------------------------------
void pic_copy(U8 *dst,U32 dst_xs,U32 dst_ys,U32 dst_bs,U8 *src,U32 src_xs,U32 src_ys,U32 src_bs)
{
U32 x0,y0,a0,l0;
U32 x1,y1,a1,l1;
U32 a; U8 c,m;
l0=1; l1=1;
if (dst_bs==1) l0=dst_xs>>3;
if (dst_bs==2) l0=dst_xs>>2;
if (dst_bs==4) l0=dst_xs>>1;
if (dst_bs==8) l0=dst_xs;
if (src_bs==1) l1=src_xs>>3;
if (src_bs==2) l1=src_xs>>2;
if (src_bs==4) l1=src_xs>>1;
if (src_bs==8) l1=src_xs;
for (a0=0;a0<dst_ys*l0;a0++) dst[a0]=0;
for (y0=0;y0<dst_ys;y0++)
{
y1=(y0*(src_ys-1))/(dst_ys-1);
a0=l0*y0;
a1=l1*y1;
for (x0=0;x0<dst_xs;x0++)
{
x1=(x0*(src_xs-1))/(dst_xs-1);
c=0;
if (src_bs==1)
{
c=src[a1+(x1>>3)];
c>>=7-(x1&7);
c&=1;
}
if (src_bs==4)
{
c=src[a1+(x1>>1)];
if (U32(x0&1)==0) c>>=4;
c&=15;
}
if (dst_bs==1)
{
c<<=7-(x0&7);
a=a0+(x0>>3);
dst[a]|=c; if (!c) dst[a]^=c;
}
if (dst_bs==4)
{
if (c) c=15;
if (U32(x0&1)==0) { c<<=4; m=0x0F; } else m=0xF0;
a=a0+(x0>>1);
dst[a]&=m;
dst[a]|=c;
}
}
}
}
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
int main(void)
{
VGA_init();
pic_copy
(
(U8*)VRAM,
VRAM_xs,
VRAM_ys,
VRAM_bs,
(U8*)pic_zilog_inside,
pic_zilog_inside_xs,
pic_zilog_inside_ys,
pic_zilog_inside_bs
);
VGA_screen();
}
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
pic_zilog_inside.h:
const U32 pic_zilog_inside_xs=640;
const U32 pic_zilog_inside_ys=480;
const U32 pic_zilog_inside_bs=1;
const U32 pic_zilog_inside[]= // hard coded image
{
0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,
0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0x80000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,
0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000001,0x80000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,
...
0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000001,0x80000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,
0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000000,0x00000001,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,
0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,0xFFFFFFFF,
};
The function pic_copy just copies hard-coded image into VRAM.
The function VGA_screen() creates the VGA signal in endless loop so other tasks must be encoded into ISRs or hard coded into pause code or between individual frames (however this is really demanding on mine setup as I got small MCU clock so there is not much room for other stuff to do). The VRAM is encoded in 16 colors (4 bits per pixel)
8 4 2 1
Brightness B G R
The brightness should just adds some voltage to R,G,B with few resistors and diodes but newer implemented it on HW side instead I have this circuit (8 colors only):
The diodes must be fast with the same barrier voltage and capacitors are 1nF. Its to avoid glitching of the image due to used interface data bus timing. Also the diodes are needed for the brightness if added in future for more info see the R2R current problem in here:
Generating square wave in AVR Assembly without PWM
[Edit1] I made huge changes in code:
//------------------------------------------------------------------------------------------------
//--- VGA EBI SDRAMC DMACA ver: 3.0 --------------------------------------------------------------
//------------------------------------------------------------------------------------------------
/*
VGA <- AT32UC3A3256
R PX10 (EBI_D0)
G PX09 (EBI_D1)
B PX08 (EBI_D2)
Bright PX07 (EBI_D3)*
/HS PX58
/VS PX59
Button PB10 (Bootloader)
debug PX54 (timing of TC00)
*/
//------------------------------------------------------------------------------------------------
//#define _Debug AVR32_PIN_PX54
#define _Button AVR32_PIN_PB10
#define _VGA_HS (1<<(AVR32_PIN_PX58&31))
#define _VGA_VS (1<<(AVR32_PIN_PX59&31))
#define _VGA_mask (_VGA_HS|_VGA_VS)
volatile avr32_gpio_port_t *port_VGA=&GPIO.port[AVR32_PIN_PX58>>5];
volatile U8 *SDRAM=(U8*)AVR32_EBI_CS0_ADDRESS;
//------------------------------------------------------------------------------------------------
//--- VGA 640x480x4 60Hz -------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
#define VRAM_xs 256
#define VRAM_ys 192
#define VRAM_bs 8
#define VRAM_ls ((VRAM_xs*VRAM_bs)>>3)
volatile static U8 VRAM[VRAM_ls*VRAM_ys];
/*------------------------------------------------------------------------------------------------
|------------------------|
| V sync |
| |----------------------|
| | V back |
| | |------------------| |
|H|H| |H|
|s|b| Video 640x480 |f|
|y|a| 525 lines |r|
|n|c| 60 Hz |o|
|c|k| |n|
| | |------------------|t|
| | V front |
|------------------------|
//----------------------------------------------------------------------------------------------*/
// VGA 640x480 60Hz H timing [pixels] dot clock = 25.175MHz
#define VGA_H_front 16
#define VGA_H_sync 96
#define VGA_H_back 48
#define VGA_H_video 640
// VGA 640x480 60Hz H timing [us] Ht = H/25.175, Hf = Vf*(VGA_V_video+VGA_V_front+VGA_V_sync+VGA_V_back)
#define VGA_Ht_front 1
#define VGA_Ht_sync 2
#define VGA_Ht_back 1
#define VGA_Hf 31500
// VGA 640x480 60Hz V timing [lines]
#define VGA_V_video 480
#define VGA_V_front 10
#define VGA_V_sync 2
#define VGA_V_back 33
#define VGA_Vf 60
//------------------------------------------------------------------------------------------------
void VGA_init();
void VGA_screen();
//------------------------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------
__attribute__((__interrupt__)) static void ISR_TC00_VGA() // TC0 chn:0 VGA horizontal frequency
{ // 8us every 31.75us -> 25% of CPU power
tc_read_sr(&AVR32_TC0,0); // more centered image requires +1 us to VGA_Ht_back -> 28% CPU power
#define y0 (VGA_V_video)
#define y1 (y0+VGA_V_front)
#define y2 (y1+VGA_V_sync)
#define y3 (y2+VGA_V_back)
#define yscr0 ((VGA_V_video>>1)-VRAM_ys)
#define yscr1 (yscr0+(VRAM_ys<<1))
static volatile U8 *p;
static volatile U32 y=y3;
#ifdef _Debug
gpio_set_gpio_pin(_Debug);
#endif
// init state
if (y>=y3){ y=0; p=VRAM; port_VGA->ovrs=_VGA_mask; }
// VS sync
if (y==y1) port_VGA->ovrc=_VGA_VS; // VS = L
if (y==y2) port_VGA->ovrs=_VGA_VS; // VS = H
// HS sync
wait_us(VGA_Ht_front); // front
port_VGA->ovrc=_VGA_HS; // HS = L
wait_us(VGA_Ht_sync); // sync
port_VGA->ovrs=_VGA_HS; // HS = H
wait_us(VGA_Ht_back); // back
// 8bit pixelformat DMACA, scan doubler + y offset
if ((y>=yscr0)&&(y<yscr1))
{
// Enable the DMACA
AVR32_DMACA.dmacfgreg = (1 << AVR32_DMACA_DMACFGREG_DMA_EN_OFFSET);
// Src Address: the source_data address
AVR32_DMACA.sar2 = (uint32_t)p;
// Dst Address: the dest_data address
AVR32_DMACA.dar2 = (uint32_t)SDRAM;
// Linked list ptrs: not used.
AVR32_DMACA.llp2 = 0x00000000;
// Channel 2 Ctrl register low
AVR32_DMACA.ctl2l =
(0 << AVR32_DMACA_CTL2L_INT_EN_OFFSET) | // Enable interrupts
(0 << AVR32_DMACA_CTL2L_DST_TR_WIDTH_OFFSET) | // Dst transfer width: 8bit (1,2 znasobi dotclock na 2x)
(0 << AVR32_DMACA_CTL2L_SRC_TR_WIDTH_OFFSET) | // Src transfer width: 8bit
(0 << AVR32_DMACA_CTL2L_DINC_OFFSET) | // Dst address increment: increment
(0 << AVR32_DMACA_CTL2L_SINC_OFFSET) | // Src address increment: increment
(0 << AVR32_DMACA_CTL2L_DST_MSIZE_OFFSET) | // Dst burst transaction len: 1 data items (each of size DST_TR_WIDTH)
(0 << AVR32_DMACA_CTL2L_SRC_MSIZE_OFFSET) | // Src burst transaction len: 1 data items (each of size DST_TR_WIDTH)
(0 << AVR32_DMACA_CTL2L_TT_FC_OFFSET) | // transfer type:M2M, flow controller: DMACA
(1 << AVR32_DMACA_CTL2L_DMS_OFFSET) | // Destination master: HSB master 2
(0 << AVR32_DMACA_CTL2L_SMS_OFFSET) | // Source master: HSB master 1
(0 << AVR32_DMACA_CTL2L_LLP_D_EN_OFFSET) | // Not used
(0 << AVR32_DMACA_CTL2L_LLP_S_EN_OFFSET); // Not used
// Channel 2 Ctrl register high
AVR32_DMACA.ctl2h =
((VRAM_ls) << AVR32_DMACA_CTL2H_BLOCK_TS_OFFSET) | // Block transfer size
(0 << AVR32_DMACA_CTL2H_DONE_OFFSET); // Not done
// Channel 2 Config register low
AVR32_DMACA.cfg2l =
(0 << AVR32_DMACA_CFG2L_HS_SEL_DST_OFFSET) | // Destination handshaking: ignored because the dst is memory
(0 << AVR32_DMACA_CFG2L_HS_SEL_SRC_OFFSET); // Source handshaking: ignored because the src is memory.
// Channel 2 Config register high
AVR32_DMACA.cfg2h =
(0 << AVR32_DMACA_CFG2H_DEST_PER_OFFSET) | // Dest hw handshaking itf: ignored because the dst is memory.
(0 << AVR32_DMACA_CFG2H_SRC_PER_OFFSET); // Source hw handshaking itf: ignored because the src is memory.
// Enable Channel 2 : start the process.
AVR32_DMACA.chenreg = ((4 << AVR32_DMACA_CHENREG_CH_EN_OFFSET) | (4 << AVR32_DMACA_CHENREG_CH_EN_WE_OFFSET));
// DMACA is messing up first BYTE so send it by SW before DMA
*SDRAM=*p;
// scan doubler increment only every second scanline
if ((y&1)==1) p+=VRAM_ls;
}
*SDRAM=0; y++;
#ifdef _Debug
gpio_clr_gpio_pin(_Debug);
#endif
#undef y0
#undef y1
#undef y2
#undef y3
}
//------------------------------------------------------------------------------------------------
void VGA_init()
{
system_init();
Disable_global_interrupt();
gpio_configure_pin(_Button,GPIO_DIR_INPUT);
#ifdef _Debug
gpio_configure_pin(_Debug,GPIO_DIR_OUTPUT);
#endif
gpio_configure_pins(port_VGA,_VGA_mask,GPIO_DIR_OUTPUT|GPIO_INIT_HIGH);
static const gpio_map_t EBI_GPIO_MAP[] =
{
{ AVR32_EBI_DATA_0_PIN, AVR32_EBI_DATA_0_FUNCTION},
{ AVR32_EBI_DATA_1_PIN, AVR32_EBI_DATA_1_FUNCTION},
{ AVR32_EBI_DATA_2_PIN, AVR32_EBI_DATA_2_FUNCTION},
{ AVR32_EBI_DATA_3_PIN, AVR32_EBI_DATA_3_FUNCTION},
};
gpio_enable_module(EBI_GPIO_MAP, sizeof(EBI_GPIO_MAP) / sizeof(EBI_GPIO_MAP[0]));
AVR32_SDRAMC.mr=0; // normal mode
AVR32_SDRAMC.tr=0; // no refresh (T=0)
AVR32_SDRAMC.cr=
(AVR32_SDRAMC_CR_NC_11_COLUMN_BITS <<AVR32_SDRAMC_CR_NC_OFFSET)
|(AVR32_SDRAMC_CR_NR_13_ROW_BITS <<AVR32_SDRAMC_CR_NR_OFFSET)
|(AVR32_SDRAMC_CR_NB_TWO_BANKS <<AVR32_SDRAMC_CR_NB_OFFSET)
|(AVR32_SDRAMC_CAS_ONE_CYCLE <<AVR32_SDRAMC_CR_CAS_OFFSET)
|(AVR32_SDRAMC_DBW_16_BITS <<AVR32_SDRAMC_CR_DBW_OFFSET)
|(0 <<AVR32_SDRAMC_TWR_OFFSET)
|(0 <<AVR32_SDRAMC_TRC_OFFSET)
|(0 <<AVR32_SDRAMC_TRP_OFFSET)
|(0 <<AVR32_SDRAMC_TRCD_OFFSET)
|(0 <<AVR32_SDRAMC_TRAS_OFFSET)
|(0 <<AVR32_SDRAMC_TXSR_OFFSET);
AVR32_SDRAMC.hsr=AVR32_SDRAMC_HSR_DA_MASK;
AVR32_SDRAMC.mdr=AVR32_SDRAMC_MDR_MD_SDRAM;
// map SDRAM CS -> memory space
AVR32_HMATRIX.sfr[AVR32_EBI_HMATRIX_NR]|=1<<AVR32_EBI_SDRAM_CS;
AVR32_HMATRIX.sfr[AVR32_EBI_HMATRIX_NR];
for (U32 a=0;a<VRAM_ls*VRAM_ys;a++) VRAM[a]=0;
timer_init(&AVR32_TC0,0,VGA_Hf,ISR_TC00_VGA,0);
Enable_global_interrupt();
}
//------------------------------------------------------------------------------------------------
Now its enough to call VGA_init(); and the stuff runs in background using Timer and DMA between internal RAM and EBI SDRAM interface. It uses only 25% of CPU power in current configuration. However half of VRAM is wasted as only 4 bits are used so high nibel might be used for back buffering to compensate. I also downclock the stuf to 66MHz as I do not have enough RAM for higher resolutions.
Related
UART bluetooth communication problem What is the proper format to send data to the UART (integer values)
I have created my functions to send and receive from the UART, and sending the data does not seem to be a problem. In the data visualizer we can see the values and even plot them. However when sending these data through the bluetooth, we cannot get the values to plot them in any of many available apps. I believe there is a problem with the way we are sending data through the UART and to the bluetooth and that is why we cannot then get the values to be plotted. Being a starter at all this, I would like to someone please advice us if the code below is ok, if there is a mistake and if there is a better way to send the data through the UART so as to make the Bluetooth work properly. Target is to be able to plot (graph) the values on the phone. Many thanks #define F_CPU 16000000UL #include <avr/io.h> #include <util/delay.h> #include <avr/interrupt.h> #include <stdio.h> #define BAUDRATE 9600 #define BAUD_PRESCALLER (((F_CPU / (BAUDRATE * 16UL))) - 1) //----------VARIABLES float V_n,V_nm1,V_measure=0; volatile int Velo_pulse; float Exp_fltr_Coeff=0.2; unsigned int Counter_ADC=0b0001; unsigned int Value1; char String[]=""; //----------Functions Definition //---timers void Timer1_Control(); void AttachInterrupt(); //---AnalogueRead void Set_Ports(); void AnalogRead_Setup(); unsigned int AnalogRead(); //---UART void USART_init(void); unsigned char USART_receive(void); void USART_send( unsigned char data); void USART_putstring(char* StringPtr, unsigned int Value1); int main(void){ USART_init(); //Call the USART initialization code Set_Ports(); AnalogRead_Setup(); AttachInterrupt(); Timer1_Control(); sei(); while(1){ _delay_ms(1); } return 0; } void USART_init(void){ UBRR0H = (unsigned char)(BAUD_PRESCALLER>>8); //UBRR0H = (uint8_t)(BAUD_PRESCALLER>>8); UBRR0L = (unsigned char)(BAUD_PRESCALLER); UCSR0B = (1<<RXEN0)|(1<<TXEN0); //Enable receiver / transmitter UCSR0C = (1<<USBS0)|(3<<UCSZ00); //Set frame format: 8data, 2stop bit } unsigned char USART_receive(void){ while(!(UCSR0A & (1<<RXC0))); //Wait for data to be received (buffer RXCn in the UCSRnA register) return UDR0; } void USART_send( unsigned char data){ while(!(UCSR0A & (1<<UDRE0))); //Waiting for empty transmit buffer (buffer UDREn in the UCSRnA register) UDR0 = data; //Loading Data on the transmit buffer } void USART_putstring(char* String, unsigned int Value1){ sprintf(String,"%d\r\n",Value1); while(*String != 0x00){ USART_send(*String); String++;} } void Set_Ports() { DDRD = 0b11111111; //All port is output DDRD ^= (1 << DDD5); // PD5 is now input } ISR(ADC_vect) { //ADMUX ^= Counter_ADC; //Swapping between ADC0 an ADC1 } void AnalogRead_Setup() { ADCSRA |= (1 << ADPS2) | (0 << ADPS1) | (0 << ADPS0); // Set ADC prescaler to 16 - 1 MHz sample rate # 16MHz ADMUX |= (1 << REFS0); // Set ADC reference to AVCC ADMUX |= (1 << ADLAR); // Left adjust ADC result to allow easy 8 bit reading ADCSRA |= (1 << ADATE); // Set ADC to Free-Running Mode ADCSRA |= (1 << ADIE); // Interrupt in Conversion Complete ADCSRA |= (1 << ADEN); // Enable ADC } unsigned int AnalogRead(unsigned int PortVal) { if (PortVal==5){ ADMUX |= (0 << MUX3) | (1 << MUX2) | (0 << MUX1) | (1 << MUX0); //sets the pin 0101 sets pin5 } else if (PortVal==4){ ADMUX |= (0 << MUX3) | (1 << MUX2) | (0 << MUX1) | (0 << MUX0); //sets the pin 0101 sets pin4 } ADCSRA |= (1 << ADSC); // Start A2D Conversions //while(ADCSRA & (1 << ADSC)); return ADCH; } //----------Timer Functions ISR (TIMER1_COMPA_vect) // Timer1 ISR (compare A vector - Compare Interrupt Mode) { cli(); V_measure=(Velo_pulse*60/0.250); //USART_putstring(String,Velo_pulse); Velo_pulse=0; V_n=Exp_fltr_Coeff*V_measure+(1-Exp_fltr_Coeff)*V_nm1; V_nm1=V_n; USART_putstring(String,(int)V_n); sei(); } ISR (INT0_vect) { Velo_pulse++; //USART_putstring(String,Velo_pulse); } void Timer1_Control() { TCCR1A=0b00000000; //Clear the timer1 registers TCCR1B=0b00000000; TCNT1=0b00000000; TCCR1B=0b00001101; //Sets prescaler (1024) & Compare mode OCR1A=2604; // 160ms - 6 Hz TIMSK1=0b00000010; } void AttachInterrupt() { DDRD ^= (1 << DDD2); // PD2 (PCINT0 pin) is now an input PORTD |= (1 << PORTD2); // turn On the Pull-up // PD2 is now an input with pull-up enabled EICRA = 0b00000011; // set INT0 to trigger on rising edge change EIMSK = 0b00000001; // Turns on INT0 }
Look at string initialization: char String[]=""; this allocates an array of chars with a size of 1 item (which is terminating zero). Then you make a call, passing this array reference as the first parameter: USART_putstring(String,(int)V_n); And the USART_putstring is as follows: void USART_putstring(char* String, unsigned int Value1){ sprintf(String,"%d\r\n",Value1); while(*String != 0x00){ USART_send(*String); String++;} } Note sprintf(String,"%d\r\n",Value1); it converts numeric value into the char buffer. I.e. the buffer should be large enough to contain the text representation of the number, line feeds \r\n\ and zero - the string terminator. But since your string buffer has size for only 1 char, it totally depends on luck, what will happen after sprintf: maybe there is some unused memory area, so the whole thing will look as if it working. Maybe there are some other variables, and their value will be overwritten, which makes the program behavior unexpected in the future. Or maybe there is some essential data, and your app will be crashing. Behavior may change after you adding several lines and recompile the code. The point is: be careful with your buffers. Instead of using constants for initialization, set the exact size for the buffer. The number length is maximum 6 symbols (1 possible sign and 5 digits, assuming you're using AVR-GCC, which has the int 16-bits wide, thus has -32768 as the minimum) + 2 for \r\n\ + 1 for terminating zero. I.e. size of the buffer should be 9 at least. char String[9];
Linux DRM ( DRI ) Cannot Screen Scrape /dev/fb0 as before with FBDEV
On other Linux machines using the FBDEV drivers ( Raspberry Pi.. etc. ), I could mmap the /dev/fb0 device and directly create a BMP file that saved what was on the screen. Now, I am trying to do the same thing with DRM on a TI Sitara AM57XX ( Beagleboard X-15 ). The code that used to work with FBDEV is shown below. This mmap no longer seems to work the DRM. I'm using a very simple Qt5 Application with the Qt platform linuxfb plugin. It draws just fine into /dev/fb0 and shows on the screen properly, however I cannot read back from /dev/fb0 with a memory mapped pointer and have an image of the screen saved to file. It looks garbled like this: Code: #ifdef FRAMEBUFFER_CAPTURE repaint(); QCoreApplication::processEvents(); // Setup framebuffer to desired format struct fb_var_screeninfo var; struct fb_fix_screeninfo finfo; memset(&finfo, 0, sizeof(finfo)); memset(&var, 0, sizeof(var)); /* Get variable screen information. Variable screen information * gives information like size of the image, bites per pixel, * virtual size of the image etc. */ int fbFd = open("/dev/fb0", O_RDWR); int fdRet = ioctl(fbFd, FBIOGET_VSCREENINFO, &var); if (fdRet < 0) { qDebug() << "Error opening /dev/fb0!"; close(fbFd); return -1; } if (ioctl(fbFd, FBIOPUT_VSCREENINFO, &var)<0) { qDebug() << "Error setting up framebuffer!"; close(fbFd); return -1; } else { qDebug() << "Success setting up framebuffer!"; } //Get fixed screen information if (ioctl(fbFd, FBIOGET_FSCREENINFO, &finfo) < 0) { qDebug() << "Error getting fixed screen information!"; close(fbFd); return -1; } else { qDebug() << "Success getting fixed screen information!"; } //int screensize = var.xres * var.yres * var.bits_per_pixel / 8; //int screensize = var.yres_virtual * finfo.line_length; //int screensize = finfo.smem_len; int screensize = finfo.line_length * var.yres_virtual; qDebug() << "Framebuffer size is: " << var.xres << var.yres << var.bits_per_pixel << screensize; int linuxFbWidth = var.xres; int linuxFbHeight = var.yres; int location = (var.xoffset) * (var.bits_per_pixel/8) + (var.yoffset) * finfo.line_length; // Perform memory mapping of linux framebuffer char* frameBufferMmapPixels = (char *)mmap(0, screensize, PROT_READ | PROT_WRITE, MAP_SHARED, fbFd, 0); assert(frameBufferMmapPixels != MAP_FAILED); QImage toSave((uchar*)frameBufferMmapPixels,linuxFbWidth,linuxFbHeight,QImage::Format_ARGB32); toSave.save("/usr/bin/test.bmp"); sync(); #endif Here is the output of the code when it runs: Success setting up framebuffer! Success getting fixed screen information! Framebuffer size is: 800 480 32 1966080 Here is the output of fbset showing the pixel format: mode "800x480" geometry 800 480 800 480 32 timings 0 0 0 0 0 0 0 accel true rgba 8/16,8/8,8/0,8/24 endmode root#am57xx-evm:~#
finfo.line_length gives the size of the actual physical scan line in bytes. It is not necessarily the same as screen width multiplied by pixel size, as scan lines may be padded. However the QImage constructor you are using assumes no padding. If xoffset is zero, it should be possible to construct a QImage directly from the framebuffer data using a constructor with the bytesPerLine argument. Otherwise there are two options: allocate a separate buffer and copy only the visible portion of each scanline to it create an image from the entire buffer (including the padding) and then crop it
If you're using DRM, then /dev/fb0 might point to an entirely different buffer (not the currently visible one) or have an different format. fbdev is really just for old legacy that hasn't been ported DRM/KMS yet and only has very limited modsetting capabilities. BTW: which kernel are you using ? hopefully not that ancient and broken TI vendor kernel ...
AVR UART - Java Android bluetooth communication
I have bluetooth module connected to AVR (Atmega32A) via UART. Some bytes that are transmit from bluetooth module to AVR are not properly recived. For example the bytes that are properly transmit/recived (UTF-8): Bluetooth module transmit byte X->recived byte X' 'w'->'w' 's'->'s' 'z'->'z' 'm'->'m' bytes recived not properly: 'q'->'y' 'p'->'~' '1'->'9' Bluetooth connection settings: Bps/Par/Bits: 115200 8N1 init UART: #define F_CLK 16000000 #define BAUD 115200 uint16_t ubrr_value = (uint16_t) (((F_CLK)/(16 * BAUD)) - 1); UBRRL = ubrr_value; UBRRH = (ubrr_value>>8); // 8 bit frame, async mode UCSRC=(1<<URSEL) | (3<<UCSZ0); //recive and transmit mode UCSRB = (1<<TXEN) | (1 << RXEN); transmit/recive byte by uart: char USART_ReceiveByte() { while(!(UCSRA & (1<<RXC))); return UDR; } void uart_sendRS(char VALUE) { while(!(UCSRA & (1<<UDRE))); UDR = VALUE; } main loop: while(1) { recivedByte = USART_ReceiveByte(); uart_sendRS(recivedByte); } i would be so glad to know why it does not work properly EDIT: if i change the order there is result: 'y'->'y' '~'->'~' '9'->'9' EDIT2: probably there is something wrong with setting UBRRL and UBRRH (ubrr_value = 7 in this case), does someone can confirm if it is proper and if the microcontroller can handle such a high BAUD? #define F_CLK 16000000 #define BAUD 115200 uint16_t ubrr_value = (uint16_t) (((F_CLK)/(16 * BAUD)) - 1); UBRRL = ubrr_value; UBRRH = (ubrr_value>>8);
The problem here is that you are not initialising the UART properly. You need to set the U2X bit in UCSRA if you wish to use the baud rate as you wish it configured. If you are using avr-libc you may use the following code to properly compute the BAUD rate. void uart0_init(void) { # define BAUD 115200 # include <util/setbaud.h> UBRRH = UBRRH_VALUE; UBRRL = UBRRL_VALUE; # if USE_2X UCSRA |= _BV(U2X); # else UCSRA &= ~_BV(U2X); # endif # undef BAUD /* other uart stuff you may need */ } If you look at the datasheet for your microcontroller, section 20.12, you will find a table with this information precomputed for you. Cheers.
What is the smallest audio buffer needed to produce Tone sound without distotions with WaveOUT API
Does the WaveOut API has some internal limitation of the size for the current piece of buffer played ? I mean if I provide a very small buffer does it affects somehow the sound played to the speakers. I am experiencing very strange noise when I am generating and playing the sinus wave with small buffer. Something like a peak, or "BUMP". The complete Story: I made a program that can generate Sinus sound signal in real time. The variable parameters are Frequency and Volume. The project requirement was to have a maximum latency of 50 ms. So the program must be able to produce Sinus signals with manually adjustable frequency of audio signal in real time. I used Windows WaveOut API, C# and P/invoke to access the API. Everything works fine when the sound buffer is 1000 ms large. If I minimize the buffer to 50 ms as per latency requirement then for certain frequencies I am experiencing at the end of every buffer, a noise or "BUMP". I do not understand if the sound generated is malformed ( I checked and is not) or something happens with the Audio chip, or some delay in initializing and playing. When I save the produced audio to .wav file everything is perfect. This means the must be some bug in my code or the audio subsystem has a limitation to the buffer chunks sent to it. For those who doesn't know WaveOut must be initialized at first time and then must be prepared with audio headers for each buffer that are containing the number of bytes that needs to be played and the pointer to a memory that contains the audio that needs to be player. UPDATE Noise happens with the following combinations 44100 SamplingRate, 16 Bits, 2 channels, 50 ms buffer and generated Sinus audio signal of 201Hz, 202Hz, 203Hz, 204Hz, 205Hz ... 219Hz, 220Hz, 240 Hz, is ok Why is this difference of 20, I do not know.
There are a few things to keep in mind when you need to output audio smoothly: waveOutXxxx API is a legacy/compatibility layer on top of lower level API and as such it has greater overhead and is not recommended when you are to reach minimal latency. Note that this is unlikely to be your primary problem, but this is a piece of general knowledge helpful for understanding because Windows is not real time OS and its audio subsystem is not realtime either you don't have control over random latency involved between you queue audio data for output and the data is really played back, the key is to keep certain level of buffer fullness which protects you from playback underflows and delivers smooth playback with waveOutXxxx you are no limited to having single buffer, you can allocate multiple reusable buffers and recycle them All in all, waveOutXxxx, DirectSound, DirectShow APIs work well with latencies 50 ms and up. With WASAPI exclusive mode streams you can get 5 ms latencies and even lower. EDIT: I seem to have said too early about 20 ms latencies. To compensate for this, here is a simple tool LowLatencyWaveOutPlay (Win32, x64) to estimate the latency you can achieve. With sufficient buffering playback is smooth, otherwise you hear stuttering. My understanding is that buffers might be returned late and the optimal design in terms of smallest latency lies along the line of having more smaller buffers so that you are given them back as early as possible. For example, 10 buffers 3 ms/buffer rather than 3 buffers 10 ms/buffer. D:\>LowLatencyWaveOutPlay.exe 48000 10 3 Format: 48000 Hz, 1 channels, 16 bits per sample Buffer Count: 10 Buffer Length: 3 ms (288 bytes) Signal Frequency: 1000 Hz ^C
So I came here because I wanted to find the basic latency of waveoutwrite() as well. I got around 25-26ms of latency before I got to the smooth sine tone. This is for: AMD Phenom(tm) 9850 Quad-Core Processor 2.51 GHz 4.00 GB ram 64-bit operating system, x64-based processor Windows 10 Enterprise N The code follows. It is a modfied version of Petzold's sine wave program, refactored to run on the command line. I also changed the polling of buffers to use of a callback on buffer complete with the idea that this would make the program more efficient, but it didn't make a difference. It also has a setup for elapsed timing, which I used to probe various timings for operations on the buffers. Using those I get: Sine wave output program Channels: 2 Sample rate: 44100 Bytes per second: 176400 Block align: 4 Bits per sample: 16 Time per buffer: 0.025850 Total time prepare header: 87.5000000000 usec Total time to fill: 327.9000000000 usec Total time for waveOutWrite: 90.8000000000 usec Program: /******************************************************************************* WaveOut example program Based on C. Petzold's sine wave example, outputs a sine wave via the waveOut API in Win32. *******************************************************************************/ #include <stdio.h> #include <windows.h> #include <math.h> #include <limits.h> #include <unistd.h> #define SAMPLE_RATE 44100 #define FREQ_INIT 440 #define OUT_BUFFER_SIZE 570*4 #define PI 3.14159 #define CHANNELS 2 #define BITS 16 #define MAXTIM 1000000000 double fAngle; LARGE_INTEGER perffreq; PWAVEHDR pWaveHdr1, pWaveHdr2; int iFreq = FREQ_INIT; VOID FillBuffer (short* pBuffer, int iFreq) { int i; int c; for (i = 0 ; i < OUT_BUFFER_SIZE ; i += CHANNELS) { for (c = 0; c < CHANNELS; c++) pBuffer[i+c] = (short)(SHRT_MAX*sin (fAngle)); fAngle += 2*PI*iFreq/SAMPLE_RATE; if (fAngle > 2 * PI) fAngle -= 2*PI; } } double elapsed(LARGE_INTEGER t) { LARGE_INTEGER rt; long tt; QueryPerformanceCounter(&rt); tt = rt.QuadPart-t.QuadPart; return (tt*(1.0/(double)perffreq.QuadPart)); } void CALLBACK waveOutProc(HWAVEOUT hwo, UINT uMsg, DWORD_PTR dwInstance, DWORD_PTR dwParam1, DWORD_PTR dwParam2) { if (uMsg == WOM_DONE) { if (pWaveHdr1->dwFlags & WHDR_DONE) { FillBuffer((short*)pWaveHdr1->lpData, iFreq); waveOutWrite(hwo, pWaveHdr1, sizeof(WAVEHDR)); } if (pWaveHdr2->dwFlags & WHDR_DONE) { FillBuffer((short*)pWaveHdr2->lpData, iFreq); waveOutWrite(hwo, pWaveHdr2, sizeof(WAVEHDR)); } } } int main() { HWAVEOUT hWaveOut ; short* pBuffer1; short* pBuffer2; short* pBuffer3; WAVEFORMATEX waveformat; UINT wReturn; int bytes; long t; LARGE_INTEGER rt; double timprep; double filtim; double waveouttim; printf("Sine wave output program\n"); fAngle = 0; /* start sine angle */ QueryPerformanceFrequency(&perffreq); pWaveHdr1 = malloc (sizeof (WAVEHDR)); pWaveHdr2 = malloc (sizeof (WAVEHDR)); pBuffer1 = malloc (OUT_BUFFER_SIZE*sizeof(short)); pBuffer2 = malloc (OUT_BUFFER_SIZE*sizeof(short)); pBuffer3 = malloc (OUT_BUFFER_SIZE*sizeof(short)); if (!pWaveHdr1 || !pWaveHdr2 || !pBuffer1 || !pBuffer2) { if (!pWaveHdr1) free (pWaveHdr1) ; if (!pWaveHdr2) free (pWaveHdr2) ; if (!pBuffer1) free (pBuffer1) ; if (!pBuffer2) free (pBuffer2) ; fprintf(stderr, "*** Error: No memory\n"); exit(1); } // Load prime parameters to format waveformat.wFormatTag = WAVE_FORMAT_PCM; waveformat.nChannels = CHANNELS; waveformat.nSamplesPerSec = SAMPLE_RATE; waveformat.wBitsPerSample = BITS; waveformat.cbSize = 0; // Calculate other parameters bytes = waveformat.wBitsPerSample/8; /* find bytes per sample */ if (waveformat.wBitsPerSample&8) bytes++; /* round up */ bytes *= waveformat.nChannels; /* find total channels size */ waveformat.nBlockAlign = bytes; /* set block align */ /* find average bytes/sec */ waveformat.nAvgBytesPerSec = bytes*waveformat.nSamplesPerSec; printf("Channels: %d\n", waveformat.nChannels); printf("Sample rate: %d\n", waveformat.nSamplesPerSec); printf("Bytes per second: %d\n", waveformat.nAvgBytesPerSec); printf("Block align: %d\n", waveformat.nBlockAlign); printf("Bits per sample: %d\n", waveformat.wBitsPerSample); printf("Time per buffer: %f\n", OUT_BUFFER_SIZE*sizeof(short)/(double)waveformat.nAvgBytesPerSec); if (waveOutOpen (&hWaveOut, WAVE_MAPPER, &waveformat, (DWORD_PTR)waveOutProc, 0, CALLBACK_FUNCTION) != MMSYSERR_NOERROR) { free (pWaveHdr1) ; free (pWaveHdr2) ; free (pBuffer1) ; free (pBuffer2) ; hWaveOut = NULL ; fprintf(stderr, "*** Error: No memory\n"); exit(1); } // Set up headers and prepare them pWaveHdr1->lpData = (LPSTR)pBuffer1; pWaveHdr1->dwBufferLength = OUT_BUFFER_SIZE*sizeof(short); pWaveHdr1->dwBytesRecorded = 0; pWaveHdr1->dwUser = 0; pWaveHdr1->dwFlags = WHDR_DONE; pWaveHdr1->dwLoops = 1; pWaveHdr1->lpNext = NULL; pWaveHdr1->reserved = 0; QueryPerformanceCounter(&rt); waveOutPrepareHeader(hWaveOut, pWaveHdr1, sizeof (WAVEHDR)); timprep = elapsed(rt); pWaveHdr2->lpData = (LPSTR)pBuffer2; pWaveHdr2->dwBufferLength = OUT_BUFFER_SIZE*sizeof(short); pWaveHdr2->dwBytesRecorded = 0; pWaveHdr2->dwUser = 0; pWaveHdr2->dwFlags = WHDR_DONE; pWaveHdr2->dwLoops = 1; pWaveHdr2->lpNext = NULL; pWaveHdr2->reserved = 0; waveOutPrepareHeader(hWaveOut, pWaveHdr2, sizeof (WAVEHDR)); // Send two buffers to waveform output device QueryPerformanceCounter(&rt); FillBuffer (pBuffer1, iFreq); filtim = elapsed(rt); QueryPerformanceCounter(&rt); waveOutWrite (hWaveOut, pWaveHdr1, sizeof (WAVEHDR)); waveouttim = elapsed(rt); FillBuffer (pBuffer2, iFreq); waveOutWrite (hWaveOut, pWaveHdr2, sizeof (WAVEHDR)); // Run waveform loop sleep(10); printf("Total time prepare header: %.10f usec\n", timprep*1000000); printf("Total time to fill: %.10f usec\n", filtim*1000000); printf("Total time for waveOutWrite: %.10f usec\n", waveouttim*1000000); waveOutUnprepareHeader(hWaveOut, pWaveHdr1, sizeof (WAVEHDR)); waveOutUnprepareHeader(hWaveOut, pWaveHdr2, sizeof (WAVEHDR)); // Close waveform file free (pWaveHdr1) ; free (pWaveHdr2) ; free (pBuffer1) ; free (pBuffer2) ; }
Pulse width modulation (PWM) on AVR Studio
I'm trying to use PWM for an LED on an ATmega8, any pin of port B. Setting up timers has been a annoying, and I don't know what to do with my OCR1A. Here's my code, and I'd love some feedback. I'm just trying to figure out how use PWM. I know the concept, and OCR1A is supposed to be the fraction of the whole counter time I want the pulse on. #define F_CPU 1000000 // 1 MHz #include <avr/io.h> #include <avr/delay.h> #include <avr/interrupt.h> int main(void){ TCCR1A |= (1 << CS10) | (1 << CS12) | (1 << CS11); OCR1A = 0x0000; TCCR1A |= ( 0 << WGM11 ) | ( 1 << WGM10 ) | (WGM12 << 1) | (WGM13 << 0); TCCR1A |= ( 1 << COM1A0 ) | ( 0 << COM1A1 ); TIMSK |= (1 << TOIE1); // Enable timer interrupt DDRB = 0xFF; sei(); // Enable global interrupts PORTB = 0b00000000; while(1) { OCR1A = 0x00FF; //I'm trying to get the timer to alternate being on for 100% of the time, _delay_ms(200); OCR1A = 0x0066; // Then 50% _delay_ms(200); OCR1A = 0x0000; // Then 0% _delay_ms(200); } } ISR (TIMER1_COMA_vect) // timer0 overflow interrupt { PORTB =~ PORTB; }
No, this is not the way how you should do a PWM. For example, how do you set a PWM rate of, for example, 42% with it? Also, the code size is big, it can be done in a much more efficient way. Also, you waste a 16 bit timer to do 8 bit operations. You have 2x 8 bit timers (Timer/Counter 0 and 2), and one 16 bit timer, Timer/Counter 1. It's also a bad idea to set unused portpins to output. All portpins which are not connected to anything, should be left as inputs. The ATmega8 has a built-in PWM generator on timers 1 and 2, there is no need in simulating it through software. You don't even have to set your ports manually (you only have to set the corresponding portpin to output) You don't even need any interrupt. #define fillrate OCR2A //... // main() PORTB=0x00; DDRB=0x08; //We use PORTB.3 as output, for OC2A, see the atmega8 reference manual // Mode: Phase correct PWM top=0xFF // OC2A output: Non-Inverted PWM TCCR2A=0x81; // Set the speed here, it will depend on your clock rate. TCCR2B=0x02; // for example, this will alternate between 75% and 42% PWM while(1) { fillrate = 191; // ca. 75% PWM delay_ms(2000); fillrate = 107; // ca. 42% PWM delay_ms(2000); } Note that you can use another LED with another PWM, by using the same timer and setting OCR2B instead of OCR2A. Don't forget to set TCCR2A to enable OCR2B as output for your PWM, as in this example only OCR2A is allowed.
You need to initialize your OCR1A with these two lines: TCCR1A = (1 << WGM10) | (1 << COM1A1); TCCR1B = (1 << CS10) | (1 << WGM12); And then use this: OCR1A = in And know that the range is 0-255. Count your percentages, and there you have it! #define F_CPU 1000000 // 1 MHz #include <avr/io.h> #include <avr/delay.h> #include <avr/interrupt.h> int main(void){ TCCR1A = (1 << WGM10) | (1 << COM1A1); TCCR1B = (1 << CS10) | (1 << WGM12); DDRB = 0xFF; sei(); // Enable global interrupts PORTB = 0b00000000; while(1) { OCR1A = 255; _delay_ms(200); OCR1A = 125; _delay_ms(200); OCR1A = 0; _delay_ms(200); } }