aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/perf_counter.h2
-rw-r--r--kernel/perf_counter.c15
-rw-r--r--kernel/sysctl.c8
3 files changed, 21 insertions, 4 deletions
diff --git a/include/linux/perf_counter.h b/include/linux/perf_counter.h
index 0fcbf34a4f73..00081d84169f 100644
--- a/include/linux/perf_counter.h
+++ b/include/linux/perf_counter.h
@@ -358,6 +358,7 @@ struct file;
358struct perf_mmap_data { 358struct perf_mmap_data {
359 struct rcu_head rcu_head; 359 struct rcu_head rcu_head;
360 int nr_pages; /* nr of data pages */ 360 int nr_pages; /* nr of data pages */
361 int nr_locked; /* nr pages mlocked */
361 362
362 atomic_t poll; /* POLL_ for wakeups */ 363 atomic_t poll; /* POLL_ for wakeups */
363 atomic_t head; /* write position */ 364 atomic_t head; /* write position */
@@ -575,6 +576,7 @@ struct perf_callchain_entry {
575extern struct perf_callchain_entry *perf_callchain(struct pt_regs *regs); 576extern struct perf_callchain_entry *perf_callchain(struct pt_regs *regs);
576 577
577extern int sysctl_perf_counter_priv; 578extern int sysctl_perf_counter_priv;
579extern int sysctl_perf_counter_mlock;
578 580
579extern void perf_counter_init(void); 581extern void perf_counter_init(void);
580 582
diff --git a/kernel/perf_counter.c b/kernel/perf_counter.c
index 6e6834e0587e..2d1342738305 100644
--- a/kernel/perf_counter.c
+++ b/kernel/perf_counter.c
@@ -44,6 +44,7 @@ static atomic_t nr_munmap_tracking __read_mostly;
44static atomic_t nr_comm_tracking __read_mostly; 44static atomic_t nr_comm_tracking __read_mostly;
45 45
46int sysctl_perf_counter_priv __read_mostly; /* do we need to be privileged */ 46int sysctl_perf_counter_priv __read_mostly; /* do we need to be privileged */
47int sysctl_perf_counter_mlock __read_mostly = 128; /* 'free' kb per counter */
47 48
48/* 49/*
49 * Lock for (sysadmin-configurable) counter reservations: 50 * Lock for (sysadmin-configurable) counter reservations:
@@ -1461,7 +1462,7 @@ static void perf_mmap_close(struct vm_area_struct *vma)
1461 1462
1462 if (atomic_dec_and_mutex_lock(&counter->mmap_count, 1463 if (atomic_dec_and_mutex_lock(&counter->mmap_count,
1463 &counter->mmap_mutex)) { 1464 &counter->mmap_mutex)) {
1464 vma->vm_mm->locked_vm -= counter->data->nr_pages + 1; 1465 vma->vm_mm->locked_vm -= counter->data->nr_locked;
1465 perf_mmap_data_free(counter); 1466 perf_mmap_data_free(counter);
1466 mutex_unlock(&counter->mmap_mutex); 1467 mutex_unlock(&counter->mmap_mutex);
1467 } 1468 }
@@ -1480,6 +1481,7 @@ static int perf_mmap(struct file *file, struct vm_area_struct *vma)
1480 unsigned long nr_pages; 1481 unsigned long nr_pages;
1481 unsigned long locked, lock_limit; 1482 unsigned long locked, lock_limit;
1482 int ret = 0; 1483 int ret = 0;
1484 long extra;
1483 1485
1484 if (!(vma->vm_flags & VM_SHARED) || (vma->vm_flags & VM_WRITE)) 1486 if (!(vma->vm_flags & VM_SHARED) || (vma->vm_flags & VM_WRITE))
1485 return -EINVAL; 1487 return -EINVAL;
@@ -1507,8 +1509,12 @@ static int perf_mmap(struct file *file, struct vm_area_struct *vma)
1507 goto unlock; 1509 goto unlock;
1508 } 1510 }
1509 1511
1510 locked = vma->vm_mm->locked_vm; 1512 extra = nr_pages /* + 1 only account the data pages */;
1511 locked += nr_pages + 1; 1513 extra -= sysctl_perf_counter_mlock >> (PAGE_SHIFT - 10);
1514 if (extra < 0)
1515 extra = 0;
1516
1517 locked = vma->vm_mm->locked_vm + extra;
1512 1518
1513 lock_limit = current->signal->rlim[RLIMIT_MEMLOCK].rlim_cur; 1519 lock_limit = current->signal->rlim[RLIMIT_MEMLOCK].rlim_cur;
1514 lock_limit >>= PAGE_SHIFT; 1520 lock_limit >>= PAGE_SHIFT;
@@ -1524,7 +1530,8 @@ static int perf_mmap(struct file *file, struct vm_area_struct *vma)
1524 goto unlock; 1530 goto unlock;
1525 1531
1526 atomic_set(&counter->mmap_count, 1); 1532 atomic_set(&counter->mmap_count, 1);
1527 vma->vm_mm->locked_vm += nr_pages + 1; 1533 vma->vm_mm->locked_vm += extra;
1534 counter->data->nr_locked = extra;
1528unlock: 1535unlock:
1529 mutex_unlock(&counter->mmap_mutex); 1536 mutex_unlock(&counter->mmap_mutex);
1530 1537
diff --git a/kernel/sysctl.c b/kernel/sysctl.c
index 8203d70928d5..3b05c2b088d2 100644
--- a/kernel/sysctl.c
+++ b/kernel/sysctl.c
@@ -920,6 +920,14 @@ static struct ctl_table kern_table[] = {
920 .mode = 0644, 920 .mode = 0644,
921 .proc_handler = &proc_dointvec, 921 .proc_handler = &proc_dointvec,
922 }, 922 },
923 {
924 .ctl_name = CTL_UNNUMBERED,
925 .procname = "perf_counter_mlock_kb",
926 .data = &sysctl_perf_counter_mlock,
927 .maxlen = sizeof(sysctl_perf_counter_mlock),
928 .mode = 0644,
929 .proc_handler = &proc_dointvec,
930 },
923#endif 931#endif
924/* 932/*
925 * NOTE: do not add new entries to this table unless you have read 933 * NOTE: do not add new entries to this table unless you have read