도와주세요!!
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <pthread.h>
#include <unistd.h>
#include <termios.h>
#include <fcntl.h>
#define PORT 9605
#define MAXDATASIZE 512
fd_set readfd;
char buf[MAXDATASIZE], message[MAXDATASIZE + 1];
char *endmessage = "/end";
void serialComm(char* message);
/* 메시지 보내기 함수 : 스레드로 돌린다.*/
void* write_message(void* sockfd)
{
int sock = (int *)sockfd;
while(1) /*무한루프*/
{
if(fgets(message, MAXDATASIZE, stdin)) /* 메시지 입력시 */
{
printf("Send : ");
sprintf(buf, "%s", message);
if (send(sock, buf, strlen(buf), 0) < 0) /* server에게 메시지 전송 */
{
printf("Write Error\n");
}
if (strstr(message, endmessage) != NULL ) /* 종료문자를 입력시 종료 */
{
printf("End\n");
close(sock);
exit(0);
}
}
}
}
int main(int argc, char *argv[])
{
int sockfd, maxfd;
struct hostent *he;
struct sockaddr_in their_addr;
pthread_t threadID; /*스레드 아이디*/
if (argc != 2)
{
fprintf(stderr,"usage: client hostname\n");
exit(1);
}
if ((he=gethostbyname(argv[1])) == NULL)
{
herror("gethostbyname");
exit(1);
}
if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Socket Error");
exit(1);
}
their_addr.sin_family = AF_INET;
their_addr.sin_port = htons(PORT);
their_addr.sin_addr = *((struct in_addr *)he->h_addr);
bzero(&(their_addr.sin_zero), 8);
if (connect(sockfd, (struct sockaddr *)&their_addr, sizeof(struct sockaddr)) == -1)
{
perror("Connect Error");
exit(1);
}
maxfd = sockfd + 1;
FD_ZERO(&readfd);
pthread_create(&threadID, NULL, write_message, (void *) sockfd); /*스레드 생성*/
printf("Connect Success! \n");
while(1)
{
FD_SET(0, &readfd);
FD_SET(sockfd, &readfd);
if(select(maxfd, &readfd, (fd_set *)0, (fd_set *)0, (struct timeval *)0) < 0)
{
printf("Select Error \n");
exit(0);
} /* select 함수 준비 */
/* server에게 메시지를 받으면 */
if (FD_ISSET(sockfd, &readfd))
{
int size;
if ((size = recv(sockfd, message, MAXDATASIZE, 0)) > 0)
{
message[size] = '\0';
serialComm(message);
printf("control command : %c", message[0]);
printf("Received : %s", message); /* 메시지를 출력 */
if(strstr(message, endmessage) != NULL ) /* 만약 종료문자 포함시 */
{
close(sockfd); /* 종료 */
printf("\n");
exit(0);
}
}
}
}
close(sockfd);
return 0;
}
void serialComm(char* message)
{
int fd; /* 통신 포트를 파일 개념으로 사용하기 위한 디스크립터*/
/* 이 파일 디스크립터를 이용하여 표준 입출력 함수를 이용 */
struct termios newtio; /* 보오율이나 stop bit 크기 등의 시리얼 통신 환경을 설정하기 위해 termios 구조체를 선언 */
char command[2];
strcpy(command, message);
fd = open( "/dev/ttySAC1", O_RDWR | O_NOCTTY | O_NONBLOCK ); /* /dev/ttySCA1 : ٸ ø */
/* com1 인 /dev/ttyS0를 사용하기 위해 open()함수를 사용.
O_RDWR은 파일 디스크립터인 ft를 읽기와 쓰기 모드로 열기 위한 지정이며
O_NOCCTY와 O_NONBLOCK는 시리얼 통신 장치에 맞추어 추가 */
memset( &newtio, 0, sizeof(newtio) );
/* 시리얼 통신환경 설정을 위한 구조체 변수 newtio 값을 0 바이트로 깨끗이 채운다. */
newtio.c_cflag = B9600;
newtio.c_cflag |= CS8; /* 데이터 비트가 8bit */
newtio.c_cflag |= CLOCAL; /* 외부 모뎀을 사용하지 않고 내부 통신 포트 사용 */
newtio.c_cflag |= CREAD; /* 쓰기는 기본, 읽기도 가능하게 */
newtio.c_iflag = IGNPAR; /* 패리티 오류 무시 */
newtio.c_oflag = 0;
newtio.c_lflag = 0;
newtio.c_cc[VTIME] = 0;
newtio.c_cc[VMIN] = 1;
tcflush (fd, TCIFLUSH );
tcsetattr(fd, TCSANOW, &newtio );
printf("serial : %s \n", command);
write( fd, command, 1);
close(fd); /* 닫기 */
}
serialComm부분을 보면 홈페이지에 나와있는 예제를 보고 만들었거든요...
newtio.c_cflag = B9600; --->이부분은 AVR보드가 사용하는 보드레이트입니다.
저희는 AVR보드에 입력 char형에 따라 모터의 동작을 하도록 하였는데요..
write함수는 포인터를 써서 인수를 넘겨주는데 char형만으로 넘겨주는 함수는 따로 없는지요..
또한 open함수 부분이 달라서 안되는 것인지.. 궁금증만 늘어나네요..
아무튼 이것을 보시고 조금이라도 아시거나 조언해 줄 부분이 있으면 답변주셨으면 감사합니다...
저희가 무선랜을 통해서 입력신호를 받아 uart포트(tx,rx)로 신호를 넘겨주는 데 이 쪽을 AVR보드와 연결에 모터를 제어하려고 하는데요.. 미니컴으로 보드가 메시지를 잘 받는지 확인하면 제대로 나오는데 외부포트로 전송이 잘 안되는 것 같습니다. 결국에는 외부포트에 작은 LED하나만 연결에 진짜 신호가 가는지 봤는데, 아주 잠깐 유심히 봐야지만 확인할 정도로 불이 깜빡하는 것을 볼 수 있었습니다. 또한 write함수를 for문을 사용해 오랫동안 돌게 하면 LED의 점등시간이 조금 늘어나는 것까지는 확인하였는 데요.. 이 이후로는 안되네요.. 도대체 뭐가 문제인지 궁금합니다. 혹시나 아시는 분 있으시면 조언 좀 해주세요.. 아무내용이나 상관 없으니 기다리겠습니다.ㅡㅡ 그럼...
그러므로 데이터를 fd로 출력하기 위해서는 포인터를 넘겨 주어야 하는데ssize_t write (int fd, const void *buf, size_t n)
변수가 포인터 변수가 아니고 char 형으로 선언하고 write()함수에서 전송하시려면
변수의 주소를 넘겨 주시면 됩니다.
아래와 같이요.
char ch_byte;
write( fd, &ch_byte, 1);
프로그램 소스를 자세히 보지는 못했습니다만,
무선 랜으로 메시지를 받을 때 마다 serialComm() 함수를 호출하고,
serialComm() 은 호출될 때마다 매번 시리얼 포트를 open() 하시고 있습니다만,
이렇게 하시면 전송 처리가 느리고 포트 열기에 실패하실 수 도 있습니다.
대신에 main() 쪽에서 시리얼 포트를 한 번 열기를 하시고,
프로그램이 종료될 때 시리얼 포트를 닫는 것이 좋습니다.
그냥 포인터로 넘겨주심 되는데 무엇이 문제인지????