blob: 5b68eaf5a7d120924d74e8d56b749c8f0a054c2d (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
|
/*
* This file is part of spark.
*
* Copyright (C) Albert Pham <http://www.sk89q.com>
* Copyright (c) lucko (Luck) <luck@lucko.me>
* Copyright (c) contributors
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package me.lucko.spark.sampler;
import me.lucko.spark.util.ThreadFinder;
import java.lang.management.ThreadInfo;
import java.lang.management.ThreadMXBean;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.regex.Pattern;
import java.util.regex.PatternSyntaxException;
import java.util.stream.Collectors;
/**
* Uses the {@link ThreadMXBean} to generate {@link ThreadInfo} instances for the threads being
* sampled.
*/
@FunctionalInterface
public interface ThreadDumper {
/**
* Generates {@link ThreadInfo} data for the sampled threads.
*
* @param threadBean the thread bean instance to obtain the data from
* @return an array of generated thread info instances
*/
ThreadInfo[] dumpThreads(ThreadMXBean threadBean);
/**
* Implementation of {@link ThreadDumper} that generates data for all threads.
*/
ThreadDumper ALL = threadBean -> threadBean.dumpAllThreads(false, false);
/**
* Implementation of {@link ThreadDumper} that generates data for a specific set of threads.
*/
final class Specific implements ThreadDumper {
private final ThreadFinder threadFinder = new ThreadFinder();
private final long[] ids;
public Specific(long[] ids) {
this.ids = ids;
}
public Specific(Set<String> names) {
Set<String> namesLower = names.stream().map(String::toLowerCase).collect(Collectors.toSet());
this.ids = this.threadFinder.getThreads()
.filter(t -> namesLower.contains(t.getName().toLowerCase()))
.mapToLong(Thread::getId)
.toArray();
}
@Override
public ThreadInfo[] dumpThreads(ThreadMXBean threadBean) {
return threadBean.getThreadInfo(this.ids, Integer.MAX_VALUE);
}
}
/**
* Implementation of {@link ThreadDumper} that generates data for a regex matched set of threads.
*/
final class Regex implements ThreadDumper {
private final ThreadFinder threadFinder = new ThreadFinder();
private final Set<Pattern> namePatterns;
private final Map<Long, Boolean> cache = new HashMap<>();
public Regex(Set<String> namePatterns) {
this.namePatterns = namePatterns.stream()
.map(regex -> {
try {
return Pattern.compile(regex);
} catch (PatternSyntaxException e) {
return null;
}
})
.filter(Objects::nonNull)
.collect(Collectors.toSet());
}
@Override
public ThreadInfo[] dumpThreads(ThreadMXBean threadBean) {
return this.threadFinder.getThreads()
.filter(thread -> {
Boolean result = this.cache.get(thread.getId());
if (result != null) {
return result;
}
for (Pattern pattern : this.namePatterns) {
if (pattern.matcher(thread.getName()).matches()) {
this.cache.put(thread.getId(), true);
return true;
}
}
this.cache.put(thread.getId(), false);
return false;
})
.map(thread -> threadBean.getThreadInfo(thread.getId()))
.filter(Objects::nonNull)
.toArray(ThreadInfo[]::new);
}
}
}
|