当线程具有相同的优先级时,混淆了SCHED_FIFO策略的theads调度

时间:2017-03-23 05:27:06

标签: c multithreading linux-kernel

我在主线程中创建了两个线程ThreadA和ThreadB,三个线程具有相同的优先级。调用的顺序是ThreadB前面的ThreadA(我无法理解)。
但是,当增加优先级主线程时,调用顺序是ThreadA前面的ThreadB(我明确指出)。 我从手册页得到这条消息如下: A call to sched_setscheduler(2), sched_setparam(2), or sched_setattr(2) will put the SCHED_FIFO (or SCHED_RR) thread identified by pid at the start of the list if it was runnable. As a consequence, it may preempt the currently running thread if it has the same priority. (POSIX.1 specifies that the thread should go to the end of the list.)

代码如下:

/*
 * critical.c
 *
 * compile with  gcc critical.c -o critical -lrt -lpthread
 *
 *  result is aaaaabbbbb when three thread have same priority 
 *  result is bbbbbaaaaa when main thread has higher priority 
 */

#include <pthread.h>    /* header file for pthreads */
#include <unistd.h>     /* header file for POSIX conformance */
#include <time.h>       /* header file for POSIX time management */
#include <sched.h>      /* header file for POSIX scheduling */
#include <stdio.h>      /* header file for standard input/outputlibrary */
#define _REENTRANT      /* macro to ensure system calls are reentrant */

void *threadA(void *);   /* predefine threadA routine */
void *threadB(void *);   /* predefine threadB routine */

pthread_t threadA_id,threadB_id,main_id;   /* thread identifiers */
pthread_attr_t attrA,attrB;        /* thread attribute structures */
struct sched_param param;          /* scheduling structure for thread attributes */

int policy=SCHED_FIFO;
int priority_min,priority_max;     /* for range of priority levels */

/* main routine */
int main()
{
  struct timespec start;
  int status;                        /* check that system calls return ok */

  clock_gettime(CLOCK_REALTIME, &start);        /* get the time   */
  printf("Start time is: %d seconds %d nano_seconds\n",start.tv_sec,start.tv_nsec);

  /* Set processor affinity */
  unsigned long mask = 1; /* use only 1 CPU core */
  unsigned int len = sizeof(mask);
  status = sched_setaffinity(0, len, &mask);
  if (status < 0) perror("sched_setaffinity");
  status = sched_getaffinity(0, len, &mask);
  if (status < 0) perror("sched_getaffinity");

  /* Find priority limits */

  priority_max = sched_get_priority_max(policy);
  priority_min = sched_get_priority_min(policy);

  /* Change priority and policy of main thread */

  main_id = pthread_self();
  param.sched_priority=priority_min;
  status = pthread_setschedparam(main_id, policy, &param);
  if (status != 0) perror("pthread_setschedparam"); /* error check */

  /* Create threadA */

  param.sched_priority = priority_min;
  pthread_attr_init(&attrA);
  status = pthread_attr_setschedpolicy(&attrA,policy);
  if (status != 0) perror("pthread_attr_setschedpolicy"); /* error check */
  status = pthread_attr_setschedparam(&attrA,&param);
  if (status != 0) perror("pthread_attr_setschedparam"); /* error check */
  status = pthread_create(&threadA_id, &attrA, threadA, NULL);
  if (status != 0) perror("pthread_create"); /* error check */
  status = pthread_setschedparam(threadA_id,policy,&param);
  if (status != 0) perror("pthread_setschedparam");

  /* Create threadB */
  param.sched_priority = priority_min; /* so that B runs with a higher priority than A */
  pthread_attr_init(&attrB);
  status = pthread_attr_setschedpolicy(&attrB,policy);
  if (status != 0) perror("pthread_attr_setschedpolicy"); /* error check */
  status = pthread_attr_setschedparam(&attrB,&param);
  if (status != 0) perror("pthread_attr_setschedparam"); /* error check */
  status = pthread_create(&threadB_id, &attrB, threadB, NULL);
  if (status != 0) perror("pthread_create"); /* error check */
  status = pthread_setschedparam(threadB_id,policy,&param);
  if (status != 0) perror("pthread_setschedparam");

  /* Join threads - force main to wait for the thread to terminate */
  printf("main() waiting for threads\n");

  status = pthread_join(threadA_id, NULL);
  if (status != 0) perror("pthread_join(threadA_id, NULL)"); /* error check */
  status = pthread_join(threadB_id, NULL);
  if (status != 0) perror("pthread_join(threadB_id, NULL)"); /* error check */

  printf("\nmain() reporting that all threads have terminated\n");
  return(0);

}  /* end of main */

void *threadA(void *arg)
{
  int j;
  for(j=1;j<=5;j++){
    printf("a");                    
  }
  return (NULL);
}

void *threadB(void *arg)
{
  int j;
  for(j=1;j<=5;j++){
    printf("b");                     
  }
  return (NULL);
}

1 个答案:

答案 0 :(得分:0)

示例是创建一个优先级为1的FIFO调度的线程:

int retVal;
pthread_attr_t attr;
struct sched_param schedParam;
pthread_t thread;

retVal = pthread_attr_init(&attr);
if (retVal)
{
    fprintf(stderr, "pthread_attr_init error %d\n", retVal);
    exit(1);
}

retVal = pthread_attr_setinheritsched(&attr, PTHREAD_EXPLICIT_SCHED);
if (retVal)
{
    fprintf(stderr, "pthread_attr_setinheritsched error %d\n", retVal);
    exit(1);
}

retVal = pthread_attr_setschedpolicy(&attr, SCHED_FIFO);
if (retVal)
{
    fprintf(stderr, "pthread_attr_setschedpolicy error %d\n", retVal);
    exit(1);
}

schedParam.sched_priority = 1;
retVal = pthread_attr_setschedparam(&attr, &schedParam);
if (retVal)
{
    fprintf(stderr, "pthread_attr_setschedparam error %d\n", retVal);
    exit(1);
}

retVal = pthread_create(&thread,
                        &attr,
                        _Thread,
                        NULL);    
if (retVal)
{
    fprintf(stderr, "pthread_create error %d\n", retVal);
    exit(1);
}

如果您的实时优先级软限制过于严格。

在运行代码之前,在shell中调用ulimit -r unlimited。或者直接在您的代码中致电setrlimit(RLIMIT_RTPRIO, ...)

系统范围限制在/etc/security/limits.conf

中设置