<LED를 켜기 위한 코드>

본 글은 Atmega128L, Osc 12Mhz 보드에서 실험하였다.

 

준비물: 

LED가 연결된 port 확인! (portA, portB, portC....),

각 port에 해당되는 Data Direction Register(DDR) 확인!

 

DDR의 각 비트는 입/출력 방향을 결정하는 비트이며, 각 비트의 입/출력은 해당되는 port의 각 pin의 입/출력을 결정짓는다.

특정 핀에 해당하는 DDR 레지스터 값이 1이면 출력 용도로 사용 할 핀임을 의미하며, 반대로 0이면 입력 용도로 사용 할 핀임을 의미한다.

 

 

Atmega 128L에서 PORTA의 주소는 

 

PORTA에서 $1B($3B)라고 hex값으로 주소가 나와있는데 괄호 밖의 주소는 Atmega103 호환모드로 사용할 때의 주소라서 괄호 안의 주소를 사용하기로 한다. 즉, 0x3B를 사용한다.

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
#include <util/delay.h>
 
#define DDRA    (*((volatile unsigned char*)0x3A))
#define PORTA    (*((volatile unsigned char*)0x3B))
#define PORTA0    0
#define PORTA1    1
#define PORTA2    2
#define PORTA3    3
#define PORTA4    4
#define PORTA5    5
#define PORTA6    6
#define PORTA7    7
 
 
int main(void)
{    
    DDRA = 0xFF;
    PORTA = (PORTA & (1<<PORTA1)) | (PORTA & (1<<PORTA2)) | (PORTA & (1<<PORTA3)) | (PORTA & (1<<PORTA4)) | (PORTA & (1<<PORTA5)) | (PORTA & (1<<PORTA6)) | (PORTA & (1<<PORTA7));
    
    while(1)
    {
        
#if 1
 
        _delay_ms(5000);
        PORTA = ~PORTA;
        
#endif
 
    }
        
    return 0;
}
 

_delay_ms함수를 사용하기 위해 #include <util/delay.h> 헤더파일을 추가했으며, 나머지는 레지스터 주소를 직접 건드리는 방식으로 코드를 사용했다.

 

PORTA에 값을 입력하면 LED가 켜지는 줄 알았지만 PORT0을 1로 설정을 안 했는데도 모든 LED가 다 켜져서 PORT0과 PORT7만 1로 설정을 하고 나머지를 모두 0으로 해보았다. 그런데도 모든 불빛이 다 들어와서 DDR 레지스터의 값을 LED0번과 LED7번만 켜지게 바꿨더니 원하는대로 동작이 되었다. PORTA는 영향을 안 미치는 줄 알았는데 while문에서 PORTA의값을 틸트(~) 시켰는데 LED가 on/off되는 것을 보면 좀 더 공부해봐야 알 것 같다. (LED가 DDR 셋팅으로만 불이 켜지는 이유가 회로도를 몰랐기 때문인데 현재 보드의 LED는 Vcc에

연결돼 있어서 PORTA의 initial value가 0이므로 ground와 비슷한 효과를 내서 LED가 켜지는 것이다. 만약 LED가 ground에 연결돼 있었다면 PORTA에 값을 넣었을 때 데이터시트

처럼 정상 작동하는 것을 확인할 수 있을 것 같다.)

새로 작성한 LED 0번과 LED 7번을 켜는 코드는 다음과 같다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
#include <util/delay.h>
 
#define DDRA    (*((volatile unsigned char*)0x3A))
#define PORTA    (*((volatile unsigned char*)0x3B))
#define PORTA0    0
#define PORTA1    1
#define PORTA2    2
#define PORTA3    3
#define PORTA4    4
#define PORTA5    5
#define PORTA6    6
#define PORTA7    7
 
 
int main(void)
{    
    DDRA = 0x81;
    PORTA = 0x00;
    //PORTA = (PORTA & (1<<PORTA0)) | (PORTA & (1<<PORTA7));
    
    while(1)
    {
        
#if 1
 
        _delay_ms(5000);
        PORTA = ~PORTA;
        
#endif
 
    }
        
    return 0;
}
 

 

'언어 > 큐브인턴(Embedded)' 카테고리의 다른 글

PINA, PORTA, DDRA 설명(LED관련)  (0) 2016.10.31
UART LED가 정상 작동을 안 하는 경우  (0) 2016.10.28
printf 함수 리턴  (0) 2016.09.30
하버드 구조 VS 폰 노이만 구조  (0) 2016.09.21
MCU, MIPS  (0) 2016.09.04
Posted by 知彼知己百戰不殆
,
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#include <stdio.h>
 
typedef struct 
{
        char a;
        char c;
        short d;
        char e;
        int b;
 
} Test_Struct;
 
int main()
{
        Test_Struct str_t;
        printf("\n[STRUCT]\n");
        printf("size=%d\n\n",sizeof(str_t));
}


char(1바이트) 2개 = 2바이트
short(2바이트) 1개 = 2바이트
char(1바이트) 1개 = 1바이트
int (4바이트) 1개 = 4바이트
= 총 9바이트

그런데 12바이트가 나왔다!


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#include <stdio.h>
 
typedef struct 
{
        char a;
        char b;
        char c;
        char d;
        int e;
 
} Test_Struct;
 
int main()
{
        Test_Struct str_t;
        printf("\n[STRUCT]\n");
        printf("size=%d\n\n",sizeof(str_t));
}


char(1바이트) 4개 = 4바이트

int(4바이트) 1개 = 4바이트

= 총 8바이트

구조체 사이즈도 8바이트가 나왔다!


위의 코드와 아래 코드의 사이즈에서 왜 차이가 나는 것일까?

이유는 패딩비트가 추가되었기 때문이다. 몇몇 컴파일러는 구조체의 필드를 메모리에 위치시킬 때 중간에 빈 공간없이 이어서 할당하는 경우도 있지만, 대부분의 컴파일러는 성능향상을 위해 CPU가 접근하기 쉬운 위치에 필드를 배치한다. 그러다보니 중간에 빈 공간이 들어가게 되는데 이것을 패딩비트라고 부른다.

32bit CPU는 메모리에서 값을 읽어올 때 한 번에 32bit씩(4Byte)를 읽어온다. 그런데 레지스터가 읽는 블록의 경계에 걸쳐 멤버 변수가 생긴다면 메모리를 읽을 때 두 개의 블록을 읽는 문제가 발생한다.

예를 들면, 구조체 안에서 char a, int b 이런 식으로 4Byte 단위로 생성을 안 했을 때 문제가 발생한다. 만약 구조체 멤버 b변수에 접근하려고 하면 어떻게 될까? 

구조체의 시작 주소에서 32bit를 읽어와도 멤버 b의 비트에 모두 접근 할 수 없다. 그래서 두 번 메모리를 읽어서 첫 번째 읽은 값에서 뒤의 24bit(char 8bit를 뺀 부분)와 두 번째 읽은 값에서 앞의 8bit를 합쳐서 멤버 b의 값을 구할 수 있다. 하지만!!!!! 한 번에 할 수 있는 일을 두 번에 걸쳐서 하는 것이기 때문에 성능저하가 발생한다.

그래서 대부분의 컴파일러는 CPU가 접근하기 쉬운 메모리 위치에 필드를 배치시켜서 패딩비트가 자동으로 들어가게 된다.

즉, char a, int b라고 선언한 부분에서는 멤버 변수 a와 b사이에는 3byte의 패딩 비트가 들어간 것이다. 이렇게 하면 메모리를 3바이트나 낭비하게 되지만 각 멤버에 접근할 때 한 번씩만 메모리를 읽으면 되기 때문에 성능저하가 발생하지 않는다.


하지만! #pragma pack(1) 이란 전처리어를 사용하면, 구조체 사이즈가 달라진다!

size=12는 pragma pack을 사용하기 전의 구조체 사이즈이고, size=9는 pragma pack을 사용했을 때의 구조체 사이즈이다.

char(1Byte) + short(2Byte) + short(2Byte) + int(4Byte) = 9Byte

하지만 pragma pack을 사용하기 전에는 char + short + padding bit = 4Byte,

short + padding bit = 4Byte,

int = 4Byte.... 그래서 12바이트가 나온 것이다.

'언어 > C' 카테고리의 다른 글

[CubeBite] File_Create_Write 함수  (0) 2017.02.08
[Tip] File I/O Error 다루는 법  (0) 2016.11.05
비트필드 구조체  (0) 2016.10.27
volatile 형한정어  (0) 2016.10.19
scanf와 fgets 함수를 같이 사용하면서 나타나는 문제점  (0) 2016.10.19
Posted by 知彼知己百戰不殆
,

비트필드 구조체

언어/C 2016. 10. 27. 17:37

요즘은 안 쓰인다고 하지만 예전에 메모리가 귀하던 시절에는 메모리를 조금이라도 아끼기 위해서 사용되었다고 한다.

1
2
3
4
5
6
7
8
9
typedef struct
{
    char a:1;
    char b:1;
    char c:1;
    char d:1;
 
} _t_TestBit;
 


이렇게 비트필드 구조체를 작성하고 sizeof로 크기를 찍어보면 1바이트가 나온다.

1
2
3
4
5
6
7
8
9
typedef struct
{
    char a:2;
    char b:2;
    char c:2;
    char d:3;
 
} _t_TestBit;
 


이렇게 비트필드 구조체를 작성하고 sizeof로 크기를 찍어보면 2바이트가 나온다. (2,2,2,2)로 작성하면 1바이트가 나온다.


char를 사용하면 최소 1바이트 부터 값이 찍히지만 int를 사용하면 4바이트부터 값이 찍힌다.

비트필드를 사용할 때는 변수명을 안 적을 수도 있는데 만약 char :0; 이라고 적으면 현재 워드(char, int)의 미사용 비트를 모두 버린다. 혹은 char :1;이런 식으로 변수명을 안 적어도 비트를 버리는 역할만 한다.

Posted by 知彼知己百戰不殆
,

volatile 형한정어

언어/C 2016. 10. 19. 10:36

변수와 관련된 모든 연산에 대해 컴파일러가 '최적화' 규칙을 적용하지 않는다. <-> const와는 반대의 역할


CPU는 빠른 연산을 위해 캐시 메모리를 사용하는데, volatile로 선언된 변수는 캐시로 처리되지 못한다. 효율은 떨어질 수 있으나 주기억장치에 정보가 읽고 쓰이는 행위는 보장 받을 수 있다. 

Posted by 知彼知己百戰不殆
,

scanf와 fgets 함수를 같이 사용하면서 나타나는 문제점


fgets 함수를 먼저 쓰든 scanf를 먼저 쓰든 두 함수가 연속적으로 온다면 둘중에 한 개의 함수만 실행되고 나머지 함수는 무시된다. fgets 함수나 scanf함수는 입력 버퍼에 Enter키가 입력될 때까지 입력을 받고, 출력은 \n(개행문자) 전까지의 데이터를 출력해준다. 따라서 입력 버퍼에는 아직 \n(개행문자)가 남아있으므로 두 번째 함수가 실행될 때는 자동으로 개행문자 데이터가 남아있으므로 함수가 입력이 끝난 줄 알고 종료한다.

따라서 이런 현상을 방지하려면!!!!!!!


윈도우: fflush(stdin)

리눅스, UNIX: fpurge(stdin)

을 사용해서 입력 버퍼에 있는 내용을 비워내야 한다. 그리고 fflush 함수는 입력 버퍼를 비우는 함수가 아니라 출력 버퍼를 비우는 함수이다. 하지만 MS 윈도우가 C 표준을 지키지 않은 덕분에 이렇게 사용도 가능하다.


호환성을 생각하는 코드를 작성해야 한다면, scanf("%d %*c", ~~~)로 작성을 해야 한다.

%*c는 문자 하나를 입력 버퍼에서 읽어온 후 그냥 버린다는 의미이다.

Posted by 知彼知己百戰不殆
,

참고 사이트들

OS/Linux 2016. 10. 13. 17:23

C 라이브러리 함수

http://forum.falinux.com

Posted by 知彼知己百戰不殆
,
1
2
3
4
5
6
7
8
#include <stdio.h>
 
int main(void)
{
    char string[]="hello world";
 
    char* ptr="hello";
}


string 배열에 들어간 hello world는 프로그램이 실행되면서 스택 메모리에 저장된 hello world이다. 따라서 값의 수정이 가능하다.


하지만 포인터 ptr에 들어간 hello는 코드 영역(상수 영역=읽기 전용)에 hello가 저장돼 있고, 그것을 가리키는 주소값을 저장해논 변수라서 읽기만 가능하다. 즉, 값의 수정이 불가능하다.

'언어 > C' 카테고리의 다른 글

volatile 형한정어  (0) 2016.10.19
scanf와 fgets 함수를 같이 사용하면서 나타나는 문제점  (0) 2016.10.19
getc() 함수를 사용했을 때의 오류  (0) 2016.09.28
함수 포인터 컴파일 오류  (0) 2016.09.26
전처리문  (0) 2016.05.02
Posted by 知彼知己百戰不殆
,

useradd -m -d [홈디렉토리경로] [사용자계정이름]


사용자 계정제거:

sudo userdel 사용자계정이름

'OS > RaspberryPi3' 카테고리의 다른 글

Vim Syntax 설정  (0) 2016.09.27
기본계정 pi 제거 후 새로운 계정 사용하기  (0) 2016.07.17
새로운 계정에 sudo permission 추가  (0) 2016.07.17
RaspberryPi3 SSH login  (0) 2016.07.17
Posted by 知彼知己百戰不殆
,

~~printf를 리턴하는 test함수 작성

int main(

{

test();

}

에러나 경고 발생 안하면 완성





#include <stdio.h>


int (*test())(const char*, ...)

{

        return printf;

}


int main()

{

        printf("return printf\n");

        test();

}



'언어 > 큐브인턴(Embedded)' 카테고리의 다른 글

UART LED가 정상 작동을 안 하는 경우  (0) 2016.10.28
LED on/off  (0) 2016.10.28
하버드 구조 VS 폰 노이만 구조  (0) 2016.09.21
MCU, MIPS  (0) 2016.09.04
Transistor Transistor Logic = TTL  (0) 2016.09.04
Posted by 知彼知己百戰不殆
,

getc() 함수를 사용했을 때의 오류


In file included from test_variable.c:1:0:

test_variable.c: In function ‘main’:

test_variable.c:13:11: error: too few arguments to function ‘_IO_getc’

   tempBuf=getc();

In file included from /usr/include/stdio.h:74:0,

                 from test_variable.c:1:

/usr/include/libio.h:434:12: note: declared here

 extern int _IO_getc (_IO_FILE *__fp);


코드:

#include <stdio.h>

#include <stdlib.h>

#include <string.h>


int main(void)

{

        int buffer_cnt;

        char rx_buffer[255]={0};

        char tempBuf;


        while(1)

        {

                tempBuf=getc();

                printf("tempBuf=%d\n",tempBuf);

                rx_buffer[buffer_cnt++]=tempBuf;

                printf("tempBuf=%d\n",tempBuf);

        }

}


getc()함수에서 괄호 안에 어디서 입력을 받을 건지를 정해주어야 한다. 입력 받을 곳을 정해주지 않아서 발생하는 오류

get(0) => stdin에서 입력을 받겠다고 적어주면 오류가 사라진다.

Posted by 知彼知己百戰不殆
,