source: rtems/cpukit/posix/src/mutexinit.c @ 2df7fcf

5
Last change on this file since 2df7fcf was 2df7fcf, checked in by Sebastian Huber <sebastian.huber@…>, on 06/14/16 at 09:45:22

posix: _POSIX_Mutex_Default_attributes

Make _POSIX_Mutex_Default_attributes constant and independent of the
scheduler instance. Use INT_MAX to indicate the default ceiling
priority.

  • Property mode set to 100644
File size: 3.7 KB
Line 
1/**
2 * @file
3 *
4 * @brief Initialize a Mutex
5 * @ingroup POSIXAPI
6 */
7
8/*
9 *  COPYRIGHT (c) 1989-2009.
10 *  On-Line Applications Research Corporation (OAR).
11 *
12 *  The license and distribution terms for this file may be
13 *  found in the file LICENSE in this distribution or at
14 *  http://www.rtems.org/license/LICENSE.
15 */
16
17#if HAVE_CONFIG_H
18#include "config.h"
19#endif
20
21#include <rtems/posix/muteximpl.h>
22#include <rtems/posix/priorityimpl.h>
23
24/**
25 * 11.3.2 Initializing and Destroying a Mutex, P1003.1c/Draft 10, p. 87
26 *
27 * NOTE:  XXX Could be optimized so all the attribute error checking
28 *            is not performed when attr is NULL.
29 */
30
31int pthread_mutex_init(
32  pthread_mutex_t           *mutex,
33  const pthread_mutexattr_t *attr
34)
35{
36  POSIX_Mutex_Control       *the_mutex;
37  const pthread_mutexattr_t *the_attr;
38  POSIX_Mutex_Protocol       protocol;
39  const Scheduler_Control   *scheduler;
40  Priority_Control           priority;
41
42  if ( attr ) the_attr = attr;
43  else        the_attr = &_POSIX_Mutex_Default_attributes;
44
45  /* Check for NULL mutex */
46  if ( !mutex )
47    return EINVAL;
48
49  /*
50   *  The POSIX specification says:
51   *
52   *  "Attempting to initialize an already initialized mutex results
53   *  in undefined behavior."
54   *
55   *  Trying to keep the caller from doing the create when *mutex
56   *  is actually a valid ID causes grief.  All it takes is the wrong
57   *  value in an uninitialized variable to make this fail.
58   *
59   *  Thus, we do not look at *mutex.
60   */
61
62  if ( !the_attr->is_initialized )
63    return EINVAL;
64
65  /*
66   *  We only support process private mutexes.
67   */
68  if ( the_attr->process_shared == PTHREAD_PROCESS_SHARED )
69    return ENOSYS;
70
71  if ( the_attr->process_shared != PTHREAD_PROCESS_PRIVATE )
72    return EINVAL;
73
74  /*
75   *  Determine the discipline of the mutex
76   */
77  switch ( the_attr->protocol ) {
78    case PTHREAD_PRIO_NONE:
79      protocol = POSIX_MUTEX_NO_PROTOCOL;
80      break;
81    case PTHREAD_PRIO_INHERIT:
82      protocol = POSIX_MUTEX_PRIORITY_INHERIT;
83      break;
84    case PTHREAD_PRIO_PROTECT:
85      protocol = POSIX_MUTEX_PRIORITY_CEILING;
86      break;
87    default:
88      return EINVAL;
89  }
90
91#if defined(_UNIX98_THREAD_MUTEX_ATTRIBUTES)
92  /*
93   *  Validate the mutex type and set appropriate SuperCore mutex
94   *  attributes.
95   */
96  switch ( the_attr->type ) {
97    case PTHREAD_MUTEX_NORMAL:
98    case PTHREAD_MUTEX_RECURSIVE:
99    case PTHREAD_MUTEX_ERRORCHECK:
100    case PTHREAD_MUTEX_DEFAULT:
101      break;
102
103    default:
104      return EINVAL;
105  }
106#endif
107
108  if ( protocol == POSIX_MUTEX_PRIORITY_CEILING ) {
109    int prio_ceiling;
110
111    scheduler = _Scheduler_Get_own( _Thread_Get_executing() );
112    prio_ceiling = the_attr->prio_ceiling;
113
114    if ( prio_ceiling == INT_MAX ) {
115      prio_ceiling = _POSIX_Priority_Get_maximum( scheduler );
116    }
117
118    if ( !_POSIX_Priority_Is_valid( prio_ceiling ) ) {
119      return EINVAL;
120    }
121
122    priority = _POSIX_Priority_To_core( prio_ceiling );
123  }
124
125  the_mutex = _POSIX_Mutex_Allocate();
126
127  if ( !the_mutex ) {
128    _Objects_Allocator_unlock();
129    return EAGAIN;
130  }
131
132  the_mutex->protocol = protocol;
133  the_mutex->is_recursive = ( the_attr->type == PTHREAD_MUTEX_RECURSIVE );
134
135  switch ( protocol ) {
136    case POSIX_MUTEX_PRIORITY_CEILING:
137      _CORE_ceiling_mutex_Initialize(
138        &the_mutex->Mutex,
139        priority
140      );
141      break;
142    default:
143      _Assert(
144        the_mutex->protocol == POSIX_MUTEX_NO_PROTOCOL
145          || the_mutex->protocol == POSIX_MUTEX_PRIORITY_INHERIT
146      );
147      _CORE_recursive_mutex_Initialize(
148        &the_mutex->Mutex.Recursive
149      );
150      break;
151  }
152
153  _Objects_Open_u32( &_POSIX_Mutex_Information, &the_mutex->Object, 0 );
154
155  *mutex = the_mutex->Object.id;
156
157  _Objects_Allocator_unlock();
158  return 0;
159}
Note: See TracBrowser for help on using the repository browser.