switch_to.h 2.6 KB

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