[질문] C thread safe 함수? 코드?

sungjun의 이미지

thread safe한 함수는 함수안에서 static변수나 global변수를 사용안하면 괜찮다고 알고 있습니다.
그런데 지금 수정하는 프로그램을 보면 mutex락을 막 쓰더라구요. 하지만 제가 정확하게 알지 못해서
이렇게 쓰는건 아닌듯 한데 확신을 가지지 못해서 이렇게 질문 올립니다.

첫번째 사례

int ts_strlen (char *a)
{
   int return_value;
 
   sem_take (&ts_string_sema); //mutex lock
   return_value = strlen (a);
   sem_give (&ts_string_sema); //mutex unlock
 
   return return_value;
}

strlen은 thread_safe한 함수로 알고 있는데 이렇게 만들어서 ts_strlen을 사용하더라구요 이런 함수들이 strcpy, strcat등 많습니다. 그냥 strlen을 사용해도 괜찮지 안나요?

두번째 사례
이건 정말 예매 하더라구요....

s_job *job;
job = mo[mailsend_id].job; // mo는 global 변수입니다.
sem_take (&job_access_sema[job->job_id]); //mutex lock
job->current_thread_num--;
sem_give (&job_access_sema[job->job_id]); //mutex unlock

job을 스레드 함수 안에서 선언 했지만 글러벌 변수안에 있는 주소변수(?)로 대체 되니
저렇게 사용하는게 맞을거 같긴한데 문제는 저 job변수가 너무 많은 곳에서 사용 된다는겁니다.
그리고 함수안에도 mutex를 lock해줘야 하는지
s_job *job;
job = mo[mailsend_id].job;
function(job);
 
void function(s_job *job)
{
	sem_take (&job_access_sema[job->job_id]);
	....
	sem_give (&job_access_sema[job->job_id]);
}

코드를 보면 일관성있게 다 해준거 같은데 너무 많이 그냥 막 한듯도 보여서
그냥 다 하다 보니 지금은 잘돌아가는데
제가 수정하면서 저도 저렇게 코드를 짜야하는지 고민이 됩니다...

thread프로그램은 어렵군요;;;

ckebabo의 이미지

첫번째 같은 경우엔 call by value가 아니기 때문입니다.
인자로 넘겨주는 포인터가 다른 쓰레드에 의해서도 참조되는 포인터값일 경우도 있을 수 있으니까요...

haze11의 이미지

ts_strlen는 strlen가 문제가 아니라, char *a 가 문제인거 같은데요.
a의 값이 변할 수 있기 때문에 mutex 써주는..

jick의 이미지

그렇게 단순하게 말할 수 없는 게, 만약 a의 값이 다른 쓰레드에 의해 변할 수 있는 상황에서 strlen을 불렀다면 그건 부르는 쪽의 잘못이죠. 부르는 쪽에서 명시적으로 "a의 값은 도중에 바뀔 수 있지만 바뀌기 전의 값이나 후의 값 중 하나만 얻을 수 있으면 돼!"라고 결정하고 설계를 했다면 모를까...

다시 말하자면 strlen을 감싸는 mutex는 매우 높은 확률로 (1) 아무 필요가 없거나 (2) 애당초 문제있는 로직을 해결하지 못하고 문제를 잠시 감추는 역할밖에 못한다는 것입니다.

sungjun의 이미지

그럼 전역변수의 strlen을 구할려면

mutex_lock();
len1 = strlen(전역변수);
mutex_unlock();
 
len2 = strlen(지역변수);

이렇게 해야 하는건가요?

strlen이 thread safe 한 함수 라도 전역변수의 길이를 구하기위해서는 저렇게 해주면 되고
지역 변수라면 그냥 strlen을 사용하면 되는건가요?

코드를 보면 무조건 ts_strlen을 사용을 해놔서..

jick의 이미지

Case 1: 전역변수가 가리키는 스트링이 바뀌지 않는 경우

이 경우 strlen을 부를 때 mutex를 걸 필요가 없습니다.

Case 2: 전역변수가 가리키는 스트링이 도중에 바뀔 수 있는 경우.

이 경우 mutex를 걸건 말건 strlen을 부르는 것 자체가 잘못입니다. 도중에 값이 바뀔 수 있는 문자열의 길이를 얻어서 어떻게 믿고 쓰겠습니까? strlen을 포함해서 문자열을 사용하는 로직 전체 단위로 락을 걸어야죠.

...따라서 어떤 경우에든 strlen 앞뒤로 mutex를 거는 것은 제가 볼 때 상당히 의심스러운 코드입니다. 코드를 짠 사람이 thread safety의 개념을 제대로 이해하지 못하고 있다는 데 오천원.

sungjun의 이미지

막힌 곳이 뚤린 느낌이네요.
thread 프로그램이 처음이고 사수도 없어서 혼자 하는데
ts_로 되어 있는 코드들이 잘 못 되있다고는 느꼈지만 확신이 안들어서 못 건들이고 있었는데
프로그램짜면서 잘못된 코드도 문제지만 쓸데 없는 코드들이 너무 많아서 힘들게 하네요.

댓글 달기

Filtered HTML

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

BBCode

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param>
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

Textile

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • You can use Textile markup to format text.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Markdown

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • Quick Tips:
    • Two or more spaces at a line's end = Line break
    • Double returns = Paragraph
    • *Single asterisks* or _single underscores_ = Emphasis
    • **Double** or __double__ = Strong
    • This is [a link](http://the.link.example.com "The optional title text")
    For complete details on the Markdown syntax, see the Markdown documentation and Markdown Extra documentation for tables, footnotes, and more.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Plain text

  • HTML 태그를 사용할 수 없습니다.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 줄과 단락은 자동으로 분리됩니다.
댓글 첨부 파일
이 댓글에 이미지나 파일을 업로드 합니다.
파일 크기는 8 MB보다 작아야 합니다.
허용할 파일 형식: txt pdf doc xls gif jpg jpeg mp3 png rar zip.
CAPTCHA
이것은 자동으로 스팸을 올리는 것을 막기 위해서 제공됩니다.