switch_to.h 3.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. /*
  2. * Low-level task switching. This is based on information published in
  3. * the Processor Abstraction Layer and the System Abstraction Layer
  4. * manual.
  5. *
  6. * Copyright (C) 1998-2003 Hewlett-Packard Co
  7. * David Mosberger-Tang <davidm@hpl.hp.com>
  8. * Copyright (C) 1999 Asit Mallick <asit.k.mallick@intel.com>
  9. * Copyright (C) 1999 Don Dugger <don.dugger@intel.com>
  10. */
  11. #ifndef _ASM_IA64_SWITCH_TO_H
  12. #define _ASM_IA64_SWITCH_TO_H
  13. #include <linux/percpu.h>
  14. struct task_struct;
  15. /*
  16. * Context switch from one thread to another. If the two threads have
  17. * different address spaces, schedule() has already taken care of
  18. * switching to the new address space by calling switch_mm().
  19. *
  20. * Disabling access to the fph partition and the debug-register
  21. * context switch MUST be done before calling ia64_switch_to() since a
  22. * newly created thread returns directly to
  23. * ia64_ret_from_syscall_clear_r8.
  24. */
  25. extern struct task_struct *ia64_switch_to (void *next_task);
  26. extern void ia64_save_extra (struct task_struct *task);
  27. extern void ia64_load_extra (struct task_struct *task);
  28. #ifdef CONFIG_VIRT_CPU_ACCOUNTING
  29. extern void ia64_account_on_switch (struct task_struct *prev, struct task_struct *next);
  30. # define IA64_ACCOUNT_ON_SWITCH(p,n) ia64_account_on_switch(p,n)
  31. #else
  32. # define IA64_ACCOUNT_ON_SWITCH(p,n)
  33. #endif
  34. #ifdef CONFIG_PERFMON
  35. DECLARE_PER_CPU(unsigned long, pfm_syst_info);
  36. # define PERFMON_IS_SYSWIDE() (__get_cpu_var(pfm_syst_info) & 0x1)
  37. #else
  38. # define PERFMON_IS_SYSWIDE() (0)
  39. #endif
  40. #define IA64_HAS_EXTRA_STATE(t) \
  41. ((t)->thread.flags & (IA64_THREAD_DBG_VALID|IA64_THREAD_PM_VALID) \
  42. || PERFMON_IS_SYSWIDE())
  43. #define __switch_to(prev,next,last) do { \
  44. IA64_ACCOUNT_ON_SWITCH(prev, next); \
  45. if (IA64_HAS_EXTRA_STATE(prev)) \
  46. ia64_save_extra(prev); \
  47. if (IA64_HAS_EXTRA_STATE(next)) \
  48. ia64_load_extra(next); \
  49. ia64_psr(task_pt_regs(next))->dfh = !ia64_is_local_fpu_owner(next); \
  50. (last) = ia64_switch_to((next)); \
  51. } while (0)
  52. #ifdef CONFIG_SMP
  53. /*
  54. * In the SMP case, we save the fph state when context-switching away from a thread that
  55. * modified fph. This way, when the thread gets scheduled on another CPU, the CPU can
  56. * pick up the state from task->thread.fph, avoiding the complication of having to fetch
  57. * the latest fph state from another CPU. In other words: eager save, lazy restore.
  58. */
  59. # define switch_to(prev,next,last) do { \
  60. if (ia64_psr(task_pt_regs(prev))->mfh && ia64_is_local_fpu_owner(prev)) { \
  61. ia64_psr(task_pt_regs(prev))->mfh = 0; \
  62. (prev)->thread.flags |= IA64_THREAD_FPH_VALID; \
  63. __ia64_save_fpu((prev)->thread.fph); \
  64. } \
  65. __switch_to(prev, next, last); \
  66. /* "next" in old context is "current" in new context */ \
  67. if (unlikely((current->thread.flags & IA64_THREAD_MIGRATION) && \
  68. (task_cpu(current) != \
  69. task_thread_info(current)->last_cpu))) { \
  70. platform_migrate(current); \
  71. task_thread_info(current)->last_cpu = task_cpu(current); \
  72. } \
  73. } while (0)
  74. #else
  75. # define switch_to(prev,next,last) __switch_to(prev, next, last)
  76. #endif
  77. #endif /* _ASM_IA64_SWITCH_TO_H */