유저<-->커널 영역 간 데이타 주고 받기
커널 프로그래밍의 경험이 없는 가운데.... 이곳 KLDP 및 몇몇 사이트, 참고서를 참고하여
임의의 상황에서 TCP Reset Packet을 전송하여 연결을 끊도록 하는 프로젝트를
수행중에 있습니다.
우선 리눅스와 연결된 TCP connection에 대한 tcp header(커널) 정보를 얻기 위해
커널<->유저간 데이터 전달을 위한 copy_to_user(), copy_from_user()를 사용하여
다음의 프로그램을 작성하였습니다.
하지만 커널<->유저 영역간의 데이터가 정상적으로 주고받지 못하는 상황에 더이상 진행을
하지 못하고 있어 고수분들의 조언을 구하고자 합니다.
작성한 코드는 다음과 같습니다.
(***유저영역 코드***)
struct sock *tcp_sock_get(u32 saddr, u16 sport, u32 daddr, u16 dport)
{
struct sock *sk;
mine = (struct mystruct *)malloc(sizeof(struct mine));
if(mine == NULL) {
printf("Out of memory\n");
exit(1);
}
mine->source_addr = saddr;
mine->source_port = sport;
mine->desti_addr = daddr;
mine->desti_port = dport;
mine->device = 2;
printf("source_addr = %x, source_port = %x, device = %d\n", mine->source_addr, mine->source_port, mine->device);
printf("desti_addr = %x, desti_port = %x\n\n", mine->desti_addr, mine->desti_port);
sk = tcp_get_sk(mine);
return sk;
}
(***커널영역 코드***)
struct sock *sys_tcp_get_sk(struct mystruct *user_buf)
{
struct sock *sk;
struct mystruct *buf;
buf = kmalloc(sizeof(struct mystruct), GFP_KERNEL);
if(buf == NULL) return(-1);
copy_from_user(buf, user_buf, sizeof(struct mystruct));
printk("src_ip = %x, src_port = %x, dif = %d\n", buf->source_addr, buf->source_port, buf->device);
printk("dst_ip = %x, dst_port = %x\n", buf->desti_addr, buf->desti_port);
sk = tcp_v4_lookup(buf->source_addr, buf->source_port, buf->desti_addr, buf->desti_port, buf->device);
....
}
이런 상황에서 커멘드창에서 ./a.out 1.1.1.1 1 2.2.2.2 2 라고 예를들어 입력을 하면
유저영역의 코드에서는 printf문에 의해
source_addr = 2020202, source_port = 200, device = 2
desti_addr = 1010101, desti_port = 100 이라고 출력이 되는데
커널쪽 메시지를 보면 printk문에 의해
src_ip = 0, src_port = 0, dif = 0
dst_ip = 0, dst_port = 0 이라고 출력이 됩니다.
이렇게 되어 tcp_v4_lookup에 올바른 값이 들어가지 못하고 NULL 포인터를 참조하게 되어
현재 세그먼트 오류가 나는 상태입니다.
많은 코멘트 부탁드리며 미리 감사드립니다 ^^;
유저 영역에서 커널
유저 영역에서 커널 영역으로 데이터를 보내 주는 부분이 없군요.
데이터 주고 받기
유저영역에서는 sk = tcp_get_sk(mine);로 커널영역 시스템콜을 호출하고
커널영역 sys_tcp_get_sk() 함수에서는
copy_from_user(buf, user_buf, sizeof(struct mystruct));를 통해
유저영역쪽 데이터를 가져오는 것 아닌가요?
우선 해당 시스템
우선 해당 시스템 콜을 시스템콜 벡터에 추가는 하셨겠지요?
궁금한 것이 있는데 현재 유저 레벨에서 하는 일을 커널에서 처리를 하시면 될듯 한데 굳이 user <--> kernel 이렇게 왔다 갔다 하시는 이유가 따로 있으신건가요? 그냥 kernel에서 tcp header를 얻어와서 비교한 후에 reset flag 처리하시고, 체크섬 하시고 해서 보내시면 될듯 한데요~
------------------------------------------------------
아직은 젊다. 모든 것을 할 수 있는 나이란 말이지.
------------------------------------------------------
아직은 젊다. 모든 것을 할 수 있는 나이란 말이지.
댓글 달기