Subversion Repositories HelenOS

Rev

Rev 2826 | Rev 2854 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | Download | RSS feed

  1. /** @addtogroup generic
  2.  * @{
  3.  */
  4.  
  5. /**
  6.  * @file
  7.  * @brief   Tdebug.
  8.  */
  9.  
  10. #include <synch/waitq.h>
  11. #include <console/klog.h>
  12. #include <udebug/udebug.h>
  13. #include <arch.h>
  14.  
  15. void udebug_stoppable_begin(void)
  16. {
  17.     int nsc;
  18.     call_t *db_call;
  19.     ipl_t ipl;
  20.  
  21.     ipl = interrupts_disable();
  22.     spinlock_lock(&TASK->lock);
  23.  
  24.     nsc = --TASK->not_stoppable_count;
  25.     db_call = TASK->debug_begin_call;
  26.  
  27.     if (TASK->dt_state == UDEBUG_TS_BEGINNING) {
  28.         klog_printf("udebug_stoppable_begin");
  29.         klog_printf(" - nsc := %d", nsc);
  30.     }
  31.  
  32.     if (TASK->dt_state == UDEBUG_TS_BEGINNING && nsc == 0) {
  33.         TASK->dt_state = UDEBUG_TS_ACTIVE;
  34.         TASK->debug_begin_call = NULL;
  35.         spinlock_unlock(&TASK->lock);
  36.         interrupts_restore(ipl);
  37.  
  38.         IPC_SET_RETVAL(db_call->data, 0);
  39.         klog_printf("udebug_stoppable_begin/ipc_answer");
  40.         ipc_answer(&TASK->answerbox, db_call);     
  41.     } else {
  42.             spinlock_unlock(&TASK->lock);
  43.         interrupts_restore(ipl);
  44.     }
  45. }
  46.  
  47. void udebug_stoppable_end(void)
  48. {
  49.     ipl_t ipl;
  50.  
  51. restart:
  52.     ipl = interrupts_disable();
  53.     spinlock_lock(&TASK->lock);
  54.  
  55.     if ((TASK->dt_state == UDEBUG_TS_BEGINNING ||
  56.         TASK->dt_state == UDEBUG_TS_ACTIVE) &&
  57.         THREAD->debug_stop == true) {
  58.         TASK->debug_begin_call = NULL;
  59.         spinlock_unlock(&TASK->lock);
  60.         interrupts_restore(ipl);
  61.  
  62.         klog_printf("udebug_stoppable_end: waitq_sleep");
  63.         waitq_sleep(&THREAD->go_wq);
  64.         goto restart;
  65.         /* must try again - have to lose stoppability atomically */
  66.     } else {
  67.         ++TASK->not_stoppable_count;
  68.         spinlock_unlock(&TASK->lock);
  69.         interrupts_restore(ipl);
  70.     }
  71. }
  72.  
  73. void udebug_syscall_event(unative_t a1, unative_t a2, unative_t a3,
  74.     unative_t a4, unative_t a5, unative_t a6, unative_t id, unative_t rc)
  75. {
  76.     call_t *call;
  77.     ipl_t ipl;
  78.  
  79.     ipl = interrupts_disable();
  80.     spinlock_lock(&THREAD->lock);
  81.  
  82.     if (THREAD->debug_active == true) {
  83.         klog_printf("udebug_syscall_event");
  84.         call = THREAD->debug_go_call;
  85.         IPC_SET_RETVAL(call->data, 0);
  86.         IPC_SET_ARG1(call->data, UDEBUG_EVENT_SYSCALL);
  87.         IPC_SET_ARG2(call->data, id);
  88.         IPC_SET_ARG3(call->data, rc);
  89.         klog_printf("udebug_syscall_event/ipc_answer");
  90.  
  91.         THREAD->syscall_args[0] = a1;
  92.         THREAD->syscall_args[1] = a2;
  93.         THREAD->syscall_args[2] = a3;
  94.         THREAD->syscall_args[3] = a4;
  95.         THREAD->syscall_args[4] = a5;
  96.         THREAD->syscall_args[5] = a6;
  97.  
  98.         /*
  99.          * Make sure debug_stop is true when going to sleep
  100.          * in case we get woken up by DEBUG_END. (At which
  101.          * point it must be back to the initial true value).
  102.          */
  103.         THREAD->debug_stop = true;
  104.         spinlock_unlock(&THREAD->lock);
  105.  
  106.         spinlock_lock(&TASK->lock);
  107.         ipc_answer(&TASK->answerbox, THREAD->debug_go_call);
  108.         spinlock_unlock(&TASK->lock);
  109.  
  110.         interrupts_restore(ipl);
  111.  
  112.         waitq_sleep(&THREAD->go_wq);
  113.     } else {
  114.         spinlock_unlock(&THREAD->lock);
  115.         interrupts_restore(ipl);
  116.     }
  117. }
  118.  
  119. /** @}
  120.  */
  121.