write 함수에 특정 String을 넣으면 그걸 받아서 처리하도록 만들고 싶은데, 엉뚱한 값이 섞여들어갑니다.
안녕하세요.
kernel에 driver를 하나 만들고, 그 device file에 특정 string을 입력하면 그 값을 받아서 처리하도록 만들려고 합니다.
예를 들어 $ echo "AAABBBCCC" > /dev/test_dev 이렇게 넣으면, 해당 함수의 write( ) 함수가 call되고 그때 입력받은 값을 받아서 처리하게 하려고 하는데,
가끔씩 입력받은 값이 "AAABBBCCC괡(초" or "connectedAAABBBCCC" 등 엉뚱한 문자열이 섞이거나 합니다.
static ssize_t test_write(struct file *filp, const char __user *ubuf, size_t count, loff_t *offp)
{
int str_len = strlen(ubuf);
char *temp = vmalloc(str_len +1);
if (temp == NULL) {
printk(KERN_ERR "%s : failed to alloc memory %dbyte.\n", __func__, count);
return count;
}
memset(temp, 0, str_len +1);
memcpy(temp, ubuf, str_len);
printk(KERN_INFO "%s", temp);
vfree(temp);
return str_len;
}
이런 식으로 구현을 했습니다.
입력 값을 확인해 보기 위해 printk( )를 했고요....
어떤 부분이 문제일까요?
음..
일반적인 문자열 처리 개념에서 보면
int str_len = strlen(offp); <- 가 맞나요?
memcpy(temp, ubuf, str_len); text 는 ubuf 에 있는데...
커널 드라이버 개념에서 보면
copyfromuser 는요???
제가 잘못 올렸네요...
strlen(offp); 가 아니라 strlen(ubuf);로 사용하고 있었습니다....
올리면서 간략하게 쓴다는게 코드를 잘못올렸네요... 수정하였습니다.
그리고 copy_from_user도 사용해 보았는데, 똑같더군요....
count가 ubuf의 길이가 아닌가요? ubuf가
count가 ubuf의 길이가 아닌가요?
ubuf가 '\0'로 끝나는 문자열이 아닌경우 strlen(ubuf);가 실제 길이보다 더 길게 나와서 그런건 아닌지요
저도 그런게 아닌가 싶었는데요..
그런데 확인을 해보면 제가 추가한 printk( )에서는 %s 로 ubuf를 출력하도록 했는데,
이 부분은 정상적인 길이로 나옵니다.
그런데 그 뒤에 쓰레기 값이 붙어 나오는 것이더라구요...
예를 들면,ubuf 값이 "BBBB" 이고 printk("%s AAA", ubuf); 라고 하면
BBBB AAA@s9fd*# 이런식입니다....
ubuf가 잘못되었으면 AAA 가 출력되기 전에 쓰레기 문자가 나와야 하는데, AAA까지 다 출력된 다음에 쓰레기 값이 출력되는 것이죠....
뭐가 문제인지 솔직히 잘 모르겠네요... 다른 곳에서 뭔가 하는 것 같지도 않고요...
이렇게 한번 해보세요.
printk(KERN_INFO "start\n");
printk(KERN_INFO "%s\n", temp);
printk(KERN_INFO "end\n");
이러면 "end" 다음에 이상한 문자열이 나오지 않을려나요?
echo 와 > 로 인한 STDOUT 쪽에 문제가 아닐지 짐작해 봅니다.
따로 test user program 을 작성해서 한번 해보시는게 좋을듯 합니다.
그리고... 오지랍인지 모르겠는데요..
copy_from_user 을 사용 하셔야 합니다.
먼저 ubuf data count 만큼 복사하고 strlen 해야합니다.
주변동료가 이런식으로 드라이버 짤까봐 겁나네요...
ps . strlen_user 란 함수도 발견 했는데 이걸 한번 사용해 보시는것도 어떨련지요.
그래도 copy_from_user 해서 복사하셔야 겠네요.
답변 감사합니다.
답변 감사합니다.
한번 해보도록 하겠습니다~
댓글 달기