source: rtems/cpukit/rtems/src/ratemon.c @ 8d0b7d96

4.104.114.84.95
Last change on this file since 8d0b7d96 was 8d0b7d96, checked in by Joel Sherrill <joel.sherrill@…>, on 12/01/95 at 22:03:55

Insert mode argument to _Watchdog_Insert removed. Now are watchdog timers
are automatically activated upon insertion.

  • Property mode set to 100644
File size: 10.3 KB
Line 
1/*
2 *  Rate Monotonic Manager
3 *
4 *
5 *  COPYRIGHT (c) 1989, 1990, 1991, 1992, 1993, 1994.
6 *  On-Line Applications Research Corporation (OAR).
7 *  All rights assigned to U.S. Government, 1994.
8 *
9 *  This material may be reproduced by or for the U.S. Government pursuant
10 *  to the copyright license under the clause at DFARS 252.227-7013.  This
11 *  notice must appear in all copies of this file and its derivatives.
12 *
13 *  $Id$
14 */
15
16#include <rtems/system.h>
17#include <rtems/rtems/status.h>
18#include <rtems/rtems/support.h>
19#include <rtems/score/isr.h>
20#include <rtems/score/object.h>
21#include <rtems/rtems/ratemon.h>
22#include <rtems/score/thread.h>
23
24/*PAGE
25 *
26 *  _Rate_monotonic_Manager_initialization
27 *
28 *  This routine initializes all Rate Monotonic Manager related
29 *  data structures.
30 *
31 *  Input parameters:
32 *    maximum_periods - number of periods timers to initialize
33 *
34 *  Output parameters:  NONE
35 *
36 *  NOTE: The Rate Monotonic Manager is built on top of the Watchdog
37 *        Handler.
38 */
39
40void _Rate_monotonic_Manager_initialization(
41  unsigned32 maximum_periods
42)
43{
44  _Objects_Initialize_information(
45    &_Rate_monotonic_Information,
46    OBJECTS_RTEMS_PERIODS,
47    FALSE,
48    maximum_periods,
49    sizeof( Rate_monotonic_Control ),
50    FALSE,
51    RTEMS_MAXIMUM_NAME_LENGTH,
52    FALSE
53  );
54}
55
56/*PAGE
57 *
58 *  rtems_rate_monotonic_create
59 *
60 *  This directive creates a rate monotonic timer and performs
61 *  some initialization.
62 *
63 *  Input parameters:
64 *    name - name of period
65 *    id   - pointer to rate monotonic id
66 *
67 *  Output parameters:
68 *    id                - rate monotonic id
69 *    RTEMS_SUCCESSFUL - if successful
70 *    error code        - if unsuccessful
71 */
72
73rtems_status_code rtems_rate_monotonic_create(
74  rtems_name    name,
75  Objects_Id   *id
76)
77{
78  Rate_monotonic_Control *the_period;
79
80  if ( !rtems_is_name_valid( name ) )
81    return RTEMS_INVALID_NAME;
82
83  _Thread_Disable_dispatch();            /* to prevent deletion */
84
85  the_period = _Rate_monotonic_Allocate();
86
87  if ( !the_period ) {
88    _Thread_Enable_dispatch();
89    return RTEMS_TOO_MANY;
90  }
91
92  the_period->owner = _Thread_Executing;
93  the_period->state = RATE_MONOTONIC_INACTIVE;
94
95  _Objects_Open( &_Rate_monotonic_Information, &the_period->Object, &name );
96
97  *id = the_period->Object.id;
98  _Thread_Enable_dispatch();
99  return RTEMS_SUCCESSFUL;
100}
101
102/*PAGE
103 *
104 *  rtems_rate_monotonic_ident
105 *
106 *  This directive returns the system ID associated with
107 *  the rate monotonic period name.
108 *
109 *  Input parameters:
110 *    name - user defined period name
111 *    id   - pointer to period id
112 *
113 *  Output parameters:
114 *    *id               - region id
115 *    RTEMS_SUCCESSFUL - if successful
116 *    error code        - if unsuccessful
117 */
118
119rtems_status_code rtems_rate_monotonic_ident(
120  rtems_name    name,
121  Objects_Id   *id
122)
123{
124  Objects_Name_to_id_errors  status;
125
126  status = _Objects_Name_to_id(
127    &_Rate_monotonic_Information,
128    &name,
129    OBJECTS_SEARCH_LOCAL_NODE,
130    id
131  );
132
133  return _Status_Object_name_errors_to_status[ status ];
134}
135
136/*PAGE
137 *
138 *  rtems_rate_monotonic_cancel
139 *
140 *  This directive allows a thread to cancel a rate monotonic timer.
141 *
142 *  Input parameters:
143 *    id - rate monotonic id
144 *
145 *  Output parameters:
146 *    RTEMS_SUCCESSFUL - if successful and caller is not the owning thread
147 *    error code        - if unsuccessful
148 */
149
150rtems_status_code rtems_rate_monotonic_cancel(
151  Objects_Id id
152)
153{
154  Rate_monotonic_Control *the_period;
155  Objects_Locations              location;
156
157  the_period = _Rate_monotonic_Get( id, &location );
158  switch ( location ) {
159    case OBJECTS_ERROR:
160      return RTEMS_INVALID_ID;
161    case OBJECTS_REMOTE:           
162      return RTEMS_INTERNAL_ERROR;  /* should never return this */
163    case OBJECTS_LOCAL:
164      if ( !_Thread_Is_executing( the_period->owner ) ) {
165        _Thread_Enable_dispatch();
166        return RTEMS_NOT_OWNER_OF_RESOURCE;
167      }
168      (void) _Watchdog_Remove( &the_period->Timer );
169      the_period->state = RATE_MONOTONIC_INACTIVE;
170      _Thread_Enable_dispatch();
171      return RTEMS_SUCCESSFUL;
172  }
173
174  return RTEMS_INTERNAL_ERROR;   /* unreached - only to remove warnings */
175}
176
177/*PAGE
178 *
179 *  rtems_rate_monotonic_delete
180 *
181 *  This directive allows a thread to delete a rate monotonic timer.
182 *
183 *  Input parameters:
184 *    id - rate monotonic id
185 *
186 *  Output parameters:
187 *    RTEMS_SUCCESSFUL - if successful
188 *    error code        - if unsuccessful
189 */
190
191rtems_status_code rtems_rate_monotonic_delete(
192  Objects_Id id
193)
194{
195  Rate_monotonic_Control *the_period;
196  Objects_Locations              location;
197
198  the_period = _Rate_monotonic_Get( id, &location );
199  switch ( location ) {
200    case OBJECTS_ERROR:
201      return RTEMS_INVALID_ID;
202    case OBJECTS_REMOTE:            /* should never return this */
203      return RTEMS_INTERNAL_ERROR;
204    case OBJECTS_LOCAL:
205      _Objects_Close( &_Rate_monotonic_Information, &the_period->Object );
206      (void) _Watchdog_Remove( &the_period->Timer );
207      the_period->state = RATE_MONOTONIC_INACTIVE;
208      _Rate_monotonic_Free( the_period );
209      _Thread_Enable_dispatch();
210      return RTEMS_SUCCESSFUL;
211  }
212
213  return RTEMS_INTERNAL_ERROR;   /* unreached - only to remove warnings */
214}
215
216/*PAGE
217 *
218 *  rtems_rate_monotonic_period
219 *
220 *  This directive allows a thread to manipulate a rate monotonic timer.
221 *
222 *  Input parameters:
223 *    id     - rate monotonic id
224 *    length - length of period (in ticks)
225 *
226 *  Output parameters:
227 *    RTEMS_SUCCESSFUL - if successful
228 *    error code        - if unsuccessful
229 */
230
231rtems_status_code rtems_rate_monotonic_period(
232  Objects_Id        id,
233  rtems_interval    length
234)
235{
236  Rate_monotonic_Control *the_period;
237  Objects_Locations       location;
238  rtems_status_code       return_value;
239
240  the_period = _Rate_monotonic_Get( id, &location );
241  switch ( location ) {
242    case OBJECTS_ERROR:
243      return RTEMS_INVALID_ID;
244    case OBJECTS_REMOTE:            /* should never return this */
245      return RTEMS_INTERNAL_ERROR;
246    case OBJECTS_LOCAL:
247      if ( !_Thread_Is_executing( the_period->owner ) ) {
248        _Thread_Enable_dispatch();
249        return RTEMS_NOT_OWNER_OF_RESOURCE;
250      }
251
252      if ( length == RTEMS_PERIOD_STATUS ) {
253        switch ( the_period->state ) {
254          case RATE_MONOTONIC_INACTIVE:
255            return_value = RTEMS_NOT_DEFINED;
256            break;
257          case RATE_MONOTONIC_ACTIVE:
258            return_value = RTEMS_SUCCESSFUL;
259            break;
260          case RATE_MONOTONIC_EXPIRED:
261            return_value = RTEMS_TIMEOUT;
262            break;
263          default:              /* unreached -- only to remove warnings */
264            return_value = RTEMS_INTERNAL_ERROR;
265            break;
266        }
267        _Thread_Enable_dispatch();
268        return( return_value );
269      }
270
271      switch ( the_period->state ) {
272        case RATE_MONOTONIC_INACTIVE:
273          the_period->state = RATE_MONOTONIC_ACTIVE;
274          _Watchdog_Initialize(
275            &the_period->Timer,
276            _Rate_monotonic_Timeout,
277            id,
278            NULL
279          );
280          _Watchdog_Insert_ticks( &the_period->Timer, length );
281          _Thread_Enable_dispatch();
282          return RTEMS_SUCCESSFUL;
283
284        case RATE_MONOTONIC_ACTIVE:
285/* following is and could be a critical section problem */
286          _Thread_Executing->Wait.id  = the_period->Object.id;
287          if ( _Rate_monotonic_Set_state( the_period ) ) {
288            _Thread_Enable_dispatch();
289            return RTEMS_SUCCESSFUL;
290          }
291         /* has expired -- fall into next case */
292        case RATE_MONOTONIC_EXPIRED:
293          the_period->state = RATE_MONOTONIC_ACTIVE;
294          _Watchdog_Insert_ticks( &the_period->Timer, length );
295          _Thread_Enable_dispatch();
296          return RTEMS_TIMEOUT;
297      }
298  }
299
300  return RTEMS_INTERNAL_ERROR;   /* unreached - only to remove warnings */
301}
302
303/*PAGE
304 *
305 * _Rate_monotonic_Set_state
306 *
307 * This kernel routine sets the STATES_WAITING_FOR_PERIOD state in
308 * the running thread's tcb if the specified period has not expired.
309 * The ready chain is adjusted if necessary.
310 *
311 * Input parameters:
312 *   the_period - pointer to period control block
313 *
314 * Output parameters:
315 *   TRUE  - if blocked successfully for period
316 *   FALSE - if period has expired
317 *
318 *  INTERRUPT LATENCY:
319 *    delete node
320 *    priority map
321 *    select heir
322 */
323
324boolean _Rate_monotonic_Set_state(
325Rate_monotonic_Control *the_period
326)
327{
328  Thread_Control *executing;
329  Chain_Control  *ready;
330  ISR_Level              level;
331  States_Control         old_state;
332
333  executing = _Thread_Executing;
334  ready     = executing->ready;
335  _ISR_Disable( level );
336
337  old_state = executing->current_state;
338
339  if ( _Rate_monotonic_Is_expired( the_period ) ) {
340    _ISR_Enable( level );
341    return( FALSE );
342  }
343
344  executing->current_state =
345             _States_Set( STATES_WAITING_FOR_PERIOD, old_state );
346
347  if ( _States_Is_ready( old_state ) ) {
348    if ( _Chain_Has_only_one_node( ready ) ) {
349      _Chain_Initialize_empty( ready );
350      _Priority_Remove_from_bit_map( &executing->Priority_map );
351      _ISR_Flash( level );
352    } else {
353      _Chain_Extract_unprotected( &executing->Object.Node );
354      _ISR_Flash( level );
355    }
356
357    if ( _Thread_Is_heir( executing ) )
358      _Thread_Calculate_heir();
359
360    _Context_Switch_necessary = TRUE;
361  }
362
363  _ISR_Enable( level );
364  return( TRUE );
365}
366
367/*PAGE
368 *
369 *  _Rate_monotonic_Timeout
370 *
371 *  This routine processes a period ending.  If the owning thread
372 *  is waiting for the period, that thread is unblocked and the
373 *  period reinitiated.  Otherwise, the period is expired.
374 *  This routine is called by the watchdog handler.
375 *
376 *  Input parameters:
377 *    id - period id
378 *
379 *  Output parameters: NONE
380 */
381
382void _Rate_monotonic_Timeout(
383  Objects_Id  id,
384  void       *ignored
385)
386{
387  Rate_monotonic_Control *the_period;
388  Objects_Locations              location;
389  Thread_Control         *the_thread;
390
391  the_period = _Rate_monotonic_Get( id, &location );
392  switch ( location ) {
393    case OBJECTS_ERROR:
394    case OBJECTS_REMOTE:  /* impossible */
395      break;
396    case OBJECTS_LOCAL:
397      the_thread = the_period->owner;
398      if ( _States_Is_waiting_for_period( the_thread->current_state ) &&
399            the_thread->Wait.id == the_period->Object.id ) {
400        _Thread_Unblock( the_thread );
401        _Watchdog_Reset( &the_period->Timer );
402      }
403      else
404        the_period->state = RATE_MONOTONIC_EXPIRED;
405      _Thread_Unnest_dispatch();
406      break;
407  }
408}
409
Note: See TracBrowser for help on using the repository browser.