18c2ecf20Sopenharmony_ci/** 28c2ecf20Sopenharmony_ci * @file init.c 38c2ecf20Sopenharmony_ci * 48c2ecf20Sopenharmony_ci * @remark Copyright 2002 OProfile authors 58c2ecf20Sopenharmony_ci * @remark Read the file COPYING 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * @author John Levon <levon@movementarian.org> 88c2ecf20Sopenharmony_ci */ 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci#include <linux/kernel.h> 118c2ecf20Sopenharmony_ci#include <linux/oprofile.h> 128c2ecf20Sopenharmony_ci#include <linux/errno.h> 138c2ecf20Sopenharmony_ci#include <linux/init.h> 148c2ecf20Sopenharmony_ci#include <linux/param.h> /* for HZ */ 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci#ifdef CONFIG_SPARC64 178c2ecf20Sopenharmony_ci#include <linux/notifier.h> 188c2ecf20Sopenharmony_ci#include <linux/rcupdate.h> 198c2ecf20Sopenharmony_ci#include <linux/kdebug.h> 208c2ecf20Sopenharmony_ci#include <asm/nmi.h> 218c2ecf20Sopenharmony_ci 228c2ecf20Sopenharmony_cistatic int profile_timer_exceptions_notify(struct notifier_block *self, 238c2ecf20Sopenharmony_ci unsigned long val, void *data) 248c2ecf20Sopenharmony_ci{ 258c2ecf20Sopenharmony_ci struct die_args *args = data; 268c2ecf20Sopenharmony_ci int ret = NOTIFY_DONE; 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci switch (val) { 298c2ecf20Sopenharmony_ci case DIE_NMI: 308c2ecf20Sopenharmony_ci oprofile_add_sample(args->regs, 0); 318c2ecf20Sopenharmony_ci ret = NOTIFY_STOP; 328c2ecf20Sopenharmony_ci break; 338c2ecf20Sopenharmony_ci default: 348c2ecf20Sopenharmony_ci break; 358c2ecf20Sopenharmony_ci } 368c2ecf20Sopenharmony_ci return ret; 378c2ecf20Sopenharmony_ci} 388c2ecf20Sopenharmony_ci 398c2ecf20Sopenharmony_cistatic struct notifier_block profile_timer_exceptions_nb = { 408c2ecf20Sopenharmony_ci .notifier_call = profile_timer_exceptions_notify, 418c2ecf20Sopenharmony_ci}; 428c2ecf20Sopenharmony_ci 438c2ecf20Sopenharmony_cistatic int timer_start(void) 448c2ecf20Sopenharmony_ci{ 458c2ecf20Sopenharmony_ci if (register_die_notifier(&profile_timer_exceptions_nb)) 468c2ecf20Sopenharmony_ci return 1; 478c2ecf20Sopenharmony_ci nmi_adjust_hz(HZ); 488c2ecf20Sopenharmony_ci return 0; 498c2ecf20Sopenharmony_ci} 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci 528c2ecf20Sopenharmony_cistatic void timer_stop(void) 538c2ecf20Sopenharmony_ci{ 548c2ecf20Sopenharmony_ci nmi_adjust_hz(1); 558c2ecf20Sopenharmony_ci unregister_die_notifier(&profile_timer_exceptions_nb); 568c2ecf20Sopenharmony_ci synchronize_rcu(); /* Allow already-started NMIs to complete. */ 578c2ecf20Sopenharmony_ci} 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_cistatic int op_nmi_timer_init(struct oprofile_operations *ops) 608c2ecf20Sopenharmony_ci{ 618c2ecf20Sopenharmony_ci if (atomic_read(&nmi_active) <= 0) 628c2ecf20Sopenharmony_ci return -ENODEV; 638c2ecf20Sopenharmony_ci 648c2ecf20Sopenharmony_ci ops->start = timer_start; 658c2ecf20Sopenharmony_ci ops->stop = timer_stop; 668c2ecf20Sopenharmony_ci ops->cpu_type = "timer"; 678c2ecf20Sopenharmony_ci printk(KERN_INFO "oprofile: Using perfctr NMI timer interrupt.\n"); 688c2ecf20Sopenharmony_ci return 0; 698c2ecf20Sopenharmony_ci} 708c2ecf20Sopenharmony_ci#endif 718c2ecf20Sopenharmony_ci 728c2ecf20Sopenharmony_ciint __init oprofile_arch_init(struct oprofile_operations *ops) 738c2ecf20Sopenharmony_ci{ 748c2ecf20Sopenharmony_ci int ret = -ENODEV; 758c2ecf20Sopenharmony_ci 768c2ecf20Sopenharmony_ci#ifdef CONFIG_SPARC64 778c2ecf20Sopenharmony_ci ret = op_nmi_timer_init(ops); 788c2ecf20Sopenharmony_ci if (!ret) 798c2ecf20Sopenharmony_ci return ret; 808c2ecf20Sopenharmony_ci#endif 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_ci return ret; 838c2ecf20Sopenharmony_ci} 848c2ecf20Sopenharmony_ci 858c2ecf20Sopenharmony_civoid oprofile_arch_exit(void) 868c2ecf20Sopenharmony_ci{ 878c2ecf20Sopenharmony_ci} 88