Rev 1018 |
Rev 1033 |
Go to most recent revision |
Blame |
Compare with Previous |
Last modification |
View Log
| RSS feed
/*
* Project: S.Ha.R.K.
*
* Coordinators:
* Giorgio Buttazzo <giorgio@sssup.it>
* Paolo Gai <pj@gandalf.sssup.it>
*
* Authors :
* Giacomo Guidi <giacomo@gandalf.sssup.it>
* (see the web pages for full authors list)
*
* ReTiS Lab (Scuola Superiore S.Anna - Pisa - Italy)
*
* http://www.sssup.it
* http://retis.sssup.it
* http://shark.sssup.it
*/
/*
* Copyright (C) 2000,2002 Paolo Gai
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
/* Interrupt Driver Module */
#include <intdrive/intdrive/intdrive.h>
#include <intdrive/intdrive/inttask.h>
#include <kernel/model.h>
#include <kernel/descr.h>
#include <kernel/var.h>
#include <kernel/func.h>
#include <tracer.h>
#include <ll/i386/64bit.h>
/*+ Status used in the level +*/
#define INTDRIVE_READY MODULE_STATUS_BASE /*+ - Ready status +*/
#define INTDRIVE_WCET_VIOLATED MODULE_STATUS_BASE+2 /*+ when wcet is finished +*/
#define INTDRIVE_IDLE MODULE_STATUS_BASE+3 /*+ to wait the replenish +*/
#define INTDRIVE_WAIT MODULE_STATUS_BASE+4 /*+ to wait the activation */
//#define INTDRIVE_DEBUG
/*+ the level redefinition for the IntDrive +*/
typedef struct {
level_des l
; /*+ the standard level descriptor +*/
TIME replenish_period
;
TIME capacity
;
TIME q_theta
;
struct timespec act_time
;
int avail
;
int replenish_timer
;
//struct timespec replenish_expires;
//int wcet_timer;
int act_number
; /*+ the activation number +*/
int flags
; /*+ the init flags... +*/
bandwidth_t U
; /*+ the used bandwidth +*/
} INTDRIVE_level_des
;
PID INTDRIVE_task
= NIL
;
/* Replenish the capacity */
static void INTDRIVE_timer
(void *arg
)
{
INTDRIVE_level_des
*lev
= (INTDRIVE_level_des
*)(arg
);
lev
->replenish_timer
= NIL
;
#ifdef INTDRIVE_DEBUG
kern_printf
("(INTD:TIMER)");
#endif
if (INTDRIVE_task
== NIL
) return;
lev
->avail
= lev
->q_theta
;
TRACER_LOGEVENT
(FTrace_EVT_user_event_0
, 0, lev
->avail
+ INT_MAX
);
switch (proc_table
[INTDRIVE_task
].
status) {
case INTDRIVE_IDLE
:
if (lev
->act_number
) {
proc_table
[INTDRIVE_task
].
status = INTDRIVE_READY
;
event_need_reschedule
();
} else {
proc_table
[INTDRIVE_task
].
status = INTDRIVE_WAIT
;
}
break;
}
}
/*static void INTDRIVE_wcet_timer(void *arg)
{
INTDRIVE_level_des *lev = (INTDRIVE_level_des *)(arg);
lev->wcet_timer = NIL;
kern_raise(XWCET_VIOLATION,INTDRIVE_task);
}*/
static PID INTDRIVE_public_scheduler
(LEVEL l
)
{
if (INTDRIVE_task
== NIL
) return NIL
;
if (proc_table
[INTDRIVE_task
].
status == INTDRIVE_READY
||
proc_table
[INTDRIVE_task
].
status == EXE
)
return INTDRIVE_task
;
else
return NIL
;
}
static int INTDRIVE_public_create
(LEVEL l
, PID p
, TASK_MODEL
*m
)
{
HARD_TASK_MODEL
*h
;
if (m
->pclass
!= HARD_PCLASS
) return -1;
if (m
->level
!= 0 && m
->level
!= l
) return -1;
h
= (HARD_TASK_MODEL
*)m
;
if (!h
->wcet
&& h
->periodicity
!= INTDRIVE
) return -1;
if (INTDRIVE_task
!= NIL
) return -1;
INTDRIVE_task
= p
;
proc_table
[INTDRIVE_task
].
wcet = h
->wcet
;
proc_table
[INTDRIVE_task
].
avail_time = h
->wcet
;
proc_table
[INTDRIVE_task
].
status = INTDRIVE_WAIT
;
proc_table
[INTDRIVE_task
].
control &= ~CONTROL_CAP
;
return 0;
}
static void INTDRIVE_public_dispatch
(LEVEL l
, PID p
, int nostop
)
{
INTDRIVE_level_des
*lev
= (INTDRIVE_level_des
*)(level_table
[l
]);
//struct timespec time;
kern_gettime
(&(lev
->act_time
));
/*TIMESPEC_ASSIGN(&time,&(lev->act_time));
ADDUSEC2TIMESPEC(proc_table[INTDRIVE_task].wcet,&time);
if (lev->flags == INTDRIVE_CHECK_WCET)
lev->wcet_timer = kern_event_post(&time,INTDRIVE_wcet_timer,(void *)lev);*/
}
static void INTDRIVE_public_epilogue
(LEVEL l
, PID p
)
{
struct timespec
time;
INTDRIVE_level_des
*lev
= (INTDRIVE_level_des
*)(level_table
[l
]);
/*if (lev->wcet_timer != NIL)
kern_event_delete(lev->wcet_timer);*/
SUBTIMESPEC
(&schedule_time
, &(lev
->act_time
), &time);
lev
->avail
-= TIMESPEC2USEC
(&time);
TRACER_LOGEVENT
(FTrace_EVT_user_event_0
, 0, lev
->avail
+ INT_MAX
);
if (proc_table
[INTDRIVE_task
].
wcet < TIMESPEC2USEC
(&time)) {
kern_raise
(XWCET_VIOLATION
,INTDRIVE_task
);
}
}
static void INTDRIVE_public_activate
(LEVEL l
, PID p
, struct timespec
*t
)
{
struct timespec acttime
;
TIME
time, delta_capacity
;
INTDRIVE_level_des
*lev
= (INTDRIVE_level_des
*)(level_table
[l
]);
if (proc_table
[INTDRIVE_task
].
status == INTDRIVE_WAIT
) {
proc_table
[INTDRIVE_task
].
status = INTDRIVE_READY
;
lev
->act_number
++;
} else {
if (proc_table
[INTDRIVE_task
].
status == INTDRIVE_IDLE
||
proc_table
[INTDRIVE_task
].
status == INTDRIVE_READY
||
proc_table
[INTDRIVE_task
].
status == EXE
) {
#ifdef INTDRIVE_DEBUG
kern_printf
("(INTD:WAIT_REC)");
#endif
lev
->act_number
++;
}
}
if (lev
->replenish_timer
== NIL
) {
delta_capacity
= lev
->q_theta
- lev
->avail
;
mul32div32to32
(delta_capacity
, MAX_BANDWIDTH
, lev
->U
, time);
kern_gettime
(&acttime
);
ADDUSEC2TIMESPEC
(time,&acttime
);
lev
->replenish_timer
= kern_event_post
(&acttime
,INTDRIVE_timer
,(void *)lev
);
/*kern_gettime(&(lev->replenish_expires));
ADDUSEC2TIMESPEC(lev->replenish_period,&(lev->replenish_expires));
lev->replenish_timer = kern_event_post(&(lev->replenish_expires),INTDRIVE_timer,(void *)lev);*/
}
}
static void INTDRIVE_public_unblock
(LEVEL l
, PID p
)
{
/* Insert task in the correct position */
proc_table
[INTDRIVE_task
].
status = INTDRIVE_READY
;
}
static void INTDRIVE_public_block
(LEVEL l
, PID p
)
{
}
static int INTDRIVE_public_message
(LEVEL l
, PID p
, void *m
)
{
INTDRIVE_level_des
*lev
= (INTDRIVE_level_des
*)(level_table
[l
]);
struct timespec
time, acttime
;
//int delta_time;
TIME delta_capacity
, delta_time
;
lev
->act_number
--;
/*if (lev->wcet_timer != NIL)
kern_event_delete(lev->wcet_timer);*/
kern_gettime
(&acttime
);
SUBTIMESPEC
(&acttime
, &(lev
->act_time
), &time);
delta_time
= TIMESPEC2USEC
(&time);
mul32div32to32
(delta_time
, (MAX_BANDWIDTH
-lev
->U
), MAX_BANDWIDTH
, delta_capacity
);
lev
->avail
-= delta_capacity
;
//lev->avail -= TIMESPEC2USEC(&time);
TRACER_LOGEVENT
(FTrace_EVT_user_event_0
, 0, lev
->avail
+ INT_MAX
);
#ifdef INTDRIVE_DEBUG
kern_printf
("(INTD:AV:%d)",(int)(lev
->avail
));
#endif
if (lev
->avail
< 0) {
proc_table
[INTDRIVE_task
].
status = INTDRIVE_IDLE
;
if (lev
->replenish_timer
!= NIL
)
kern_event_delete
(lev
->replenish_timer
);
delta_capacity
= lev
->q_theta
- lev
->avail
;
mul32div32to32
(delta_capacity
, MAX_BANDWIDTH
, lev
->U
, delta_time
);
kern_gettime
(&acttime
);
ADDUSEC2TIMESPEC
(delta_time
,&acttime
);
lev
->replenish_timer
= kern_event_post
(&acttime
,INTDRIVE_timer
,(void *)lev
);
/*temp = -lev->avail;
mul32div32to32(temp,lev->replenish_period,lev->capacity,delta_time)
ADDUSEC2TIMESPEC(delta_time,&(lev->replenish_expires));
lev->replenish_timer = kern_event_post(&(lev->replenish_expires),INTDRIVE_timer,(void *)lev);*/
#ifdef INTDRIVE_DEBUG
kern_printf
("(INTD:IDLE:%d)",delta_time
);
#endif
} else {
if (lev
->act_number
) {
proc_table
[INTDRIVE_task
].
status = INTDRIVE_READY
;
#ifdef INTDRIVE_DEBUG
kern_printf
("(INTD:NEXT_ACT)");
#endif
} else {
#ifdef INTDRIVE_DEBUG
kern_printf
("(INTD:WAIT_ACT)");
#endif
proc_table
[INTDRIVE_task
].
status = INTDRIVE_WAIT
;
}
}
TRACER_LOGEVENT
(FTrace_EVT_task_end_cycle
, (unsigned short int)proc_table
[INTDRIVE_task
].
context,(unsigned int)l
);
return 0;
}
static void INTDRIVE_public_end
(LEVEL l
, PID p
)
{
INTDRIVE_level_des
*lev
= (INTDRIVE_level_des
*)(level_table
[l
]);
if (lev
->replenish_timer
!= NIL
)
kern_event_delete
(lev
->replenish_timer
);
/*if (lev->wcet_timer != NIL)
kern_event_delete(lev->wcet_timer);*/
proc_table
[INTDRIVE_task
].
status = INTDRIVE_IDLE
;
}
/* Registration functions */
/*+ Registration function: +*/
LEVEL INTDRIVE_register_level
(TIME capacity
, TIME q_theta
, int U
, int flags
)
{
LEVEL l
; /* the level that we register */
INTDRIVE_level_des
*lev
;
printk
("INTDRIVE_register_level\n");
/* request an entry in the level_table */
l
= level_alloc_descriptor
(sizeof(INTDRIVE_level_des
));
lev
= (INTDRIVE_level_des
*)level_table
[l
];
lev
->l.
public_scheduler = INTDRIVE_public_scheduler
;
lev
->l.
public_guarantee = NULL
;
lev
->l.
public_create = INTDRIVE_public_create
;
lev
->l.
public_end = INTDRIVE_public_end
;
lev
->l.
public_dispatch = INTDRIVE_public_dispatch
;
lev
->l.
public_epilogue = INTDRIVE_public_epilogue
;
lev
->l.
public_activate = INTDRIVE_public_activate
;
lev
->l.
public_unblock = INTDRIVE_public_unblock
;
lev
->l.
public_block = INTDRIVE_public_block
;
lev
->l.
public_message = INTDRIVE_public_message
;
NULL_TIMESPEC
(&(lev
->act_time
));
lev
->capacity
= capacity
;
lev
->replenish_timer
= NIL
;
lev
->flags
= flags
;
lev
->act_number
= 0;
lev
->avail
= 0;
lev
->q_theta
= q_theta
;
lev
->U
= U
;
//mul32div32to32(MAX_BANDWIDTH,U,10000,lev->U);
#ifdef INTDRIVE_DEBUG
printk
("INTDRIVE Init: %d %d %d\n", lev
->capacity
, lev
->q_theta
, lev
->U
);
#endif
//TODO !!! Fixing 10000 is not a clean way to work !!!
if (intdrive_taskinit
(l
, 10000) == -1)
printk
("Error: INTDRIVE task not activated!\n");
return l
;
}
bandwidth_t INTDRIVE_usedbandwidth
(LEVEL l
)
{
INTDRIVE_level_des
*lev
= (INTDRIVE_level_des
*)(level_table
[l
]);
return lev
->U
;
}
TIME INTDRIVE_set_q_theta
(LEVEL l
, TIME new_q_theta
)
{
INTDRIVE_level_des
*lev
= (INTDRIVE_level_des
*)(level_table
[l
]);
lev
->q_theta
= new_q_theta
;
if (lev
->q_theta
< 0) lev
->q_theta
= 0;
if (lev
->q_theta
> lev
->capacity
) lev
->q_theta
= lev
->capacity
;
return lev
->q_theta
;
}