JPDA 架构研究6 - Agent利用环境指针访问VM (线程管理篇)

引入:

上篇文章讲解了分类:内存管理,现在讲线程类操作的接口。

分类2:线程类操作

a.GetThreadState.获取线程状态

jvmtiError
GetThreadState(jvmtiEnv* env,
            jthread thread,
            jint* thread_state_ptr)

大家都知道线程有很多种状态,比如Alive,Terminated,Runnable, 等待进入Synchronize Block,Waiting,Sleeping,Parked,Suspended,Interrupted等。

它入参thread表示要查询的线程,返回thread_state_ptr表示线程状态。

b.GetAllThreads.获取所有活着的线程,这些线程必须是连接到当前VM并且Alive的。

jvmtiError
GetAllThreads(jvmtiEnv* env,
            jint* threads_count_ptr,
            jthread** threads_ptr)

它返回threads_count_ptr表示活着的线程数量,返回threads_ptr表示活着的线程的引用数组。

c.SuspendThread.挂起某线程

jvmtiError
SuspendThread(jvmtiEnv* env,
            jthread thread)

一旦挂起某线程,则对应方法无法返回直到另外有某个线程调用ResumeThread.

它入参thread表示要挂起的线程。

d.SuspendThreadList.挂起一组线程

jvmtiError
SuspendThreadList(jvmtiEnv* env,
            jint request_count,
            const jthread* request_list,
            jvmtiError* results)

一旦在ThreadList中某线程被挂起,则只有其他某线程调用ResumeThreadList或者ResumeThread后,此线程对应方法才可以返回。

入参request_count表示要挂起的线程数量,request_list,表示要挂起的线程数组,返回results表示挂起结果。

e.ResumeThread.恢复某个被挂起的线程

jvmtiError
ResumeThread(jvmtiEnv* env,
            jthread thread)

f.ResumeThreadList.恢复某个被挂起的线程组

jvmtiError
ResumeThreadList(jvmtiEnv* env,
            jint request_count,
            const jthread* request_list,
            jvmtiError* results)

g.StopThread.杀死某线程

jvmtiError
StopThread(jvmtiEnv* env,
            jthread thread,
            jobject exception)

h.InterruptThread.中断某线程

vmtiError
InterruptThread(jvmtiEnv* env,
            jthread thread)

i.GetThreadInfo.获取某线程信息

typedef struct {
    char* name;
    jint。 priority;
    jboolean is_daemon;
    jthreadGroup thread_group;
    jobject context_class_loader;
} jvmtiThreadInfo;
jvmtiError
GetThreadInfo(jvmtiEnv* env,
            jthread thread,
            jvmtiThreadInfo* info_ptr)

从这里可以看出,这里可以获取线程的名字,优先级,是否守护线程,所属线程组,上下文加载器等信息。

j.GetOwnerMonitorInfo.获取线程拥有的监视器(可以多个)信息

jvmtiError
GetOwnedMonitorInfo(jvmtiEnv* env,
            jthread thread,
            jint* owned_monitor_count_ptr,
            jobject** owned_monitors_ptr)

我在这里对Monitor的理解是线程持有的锁,也就是synchronized所关联的对象。

入参仍是给定线程,返回监视器的数量和监视器引用的数组。

k.GetCurrentContendedMonitor.获取线程当前的监视器。

jvmtiError
GetCurrentContendedMonitor(jvmtiEnv* env,
            jthread thread,
            jobject* monitor_ptr)

和上面配套使用。因为多个竞争者同时共享某线程,那么肯定有某个当前竞争者占用了此线程的执行。

l.SetThreadLocalStorage.VM设置一个thread-local的值来关联到某 环境-线程对。

jvmtiError
SetThreadLocalStorage(jvmtiEnv* env,
            jthread thread,
            const void* data)

m.GetThreadLocalStorage.Agent来获取为某线程设置的thread-local值。

jvmtiError
GetThreadLocalStorage(jvmtiEnv* env,
            jthread thread,
            void** data_ptr)
时间: 2024-11-03 21:32:17

JPDA 架构研究6 - Agent利用环境指针访问VM (线程管理篇)的相关文章

JPDA 架构研究9 - Agent利用环境指针访问VM (堆管理篇)

引入: 上文中我们讨论了Agent利用环境指针访问VM的堆栈操作部分,这里我们讨论堆操作部分. 分类5:堆操作 a. GetTag. 获取对象在堆上的唯一标识符(长整型). jvmtiError GetTag(jvmtiEnv* env,             jobject object,             jlong* tag_ptr) b.SetTag.为对象设置堆上的唯一标示符. jvmtiError SetTag(jvmtiEnv* env,             jobje

JPDA 架构研究10 - Agent利用环境指针访问VM(局部变量管理篇)

引入: 上篇我们讲解了Agent如何利用环境指针访问VM的管理堆的操作.这里主要讲解如何管理局部变量. 分类6:局部变量管理 a.GetLocalObject. 获取局部对象 jvmtiError GetLocalObject(jvmtiEnv* env,             jthread thread,             jint depth,             jint slot,             jobject* value_ptr) b.GetLocalInt.

JPDA 架构研究13 - Agent利用环境指针访问VM(类管理篇)

引入: 上文中提到Agent如何利用环境指针访问VM的(Watch)功能,这里主要讲解如何去管理类的. 分类9:管理类 a.GetLoadedClasses. 获得虚拟机中所有被加载的类的数组. jvmtiError GetLoadedClasses(jvmtiEnv* env,             jint* class_count_ptr,               jclass** classes_ptr) 从返回值可以看出,class_count_ptr表示被加载的类的数量,cla

JPDA 架构研究14 - Agent利用环境指针访问VM(对象管理篇)

引入: 上文讨论了Agent如何利用环境指针访问VM(管理类),这里讨论如何管理对象. 分类9:对象管理 a.GetObjectSize.  获取对象的大小, jvmtiError GetObjectSize(jvmtiEnv* env,             jobject object,             jlong* size_ptr) 注意,这不是真实大小,这只是从对象的定义代码中获得的一个预估值. b.GetObjectHashCode.获取对象的hash值. jvmtiErr

JPDA 架构研究5 - Agent利用环境指针访问VM (内存管理篇)

引入: 我们在前面说到JVMTI的客户端Agent,又提到Agent通过环境指针来访问VM.这里就来看看环境指针到底有多大的访问VM的能力. 分类1:内存管理 a.Allocate. 分配内存 jvmtiError Allocate(jvmtiEnv* env,             jlong size,             unsigned char** mem_ptr) size:分配的字节数. mem_ptr:分配内存的首地址. b.Deallocate.释放内存,释放已经分配的内

JPDA 架构研究7 - Agent利用环境指针访问VM(线程组管理篇)

引入: 上篇文章中我们讨论了Agent利用环境指针访问VM的线程操作,这里讨论线程组操作. 分类3:线程组操作 a.GetTopThreadGroups.让Agent获取VM中的所有全局的线程组. jvmtiError GetTopThreadGroups(jvmtiEnv* env,             jint* group_count_ptr,             jthreadGroup** groups_ptr) 函数会返回全局的线程组的数量和线程组的列表. b.GetThre

JPDA 架构研究8 - Agent利用环境指针访问VM(堆栈管理篇)

引入: 上篇文章讲解了Agent利用环境指针访问VM的线程组操作,这里讨论下堆栈操作. 分类4:堆栈操作 a. GetStackTrace.获取某线程的堆栈. jvmtiError GetStackTrace(jvmtiEnv* env,             jthread thread,             jint start_depth,             jint max_frame_count,             jvmtiFrameInfo* frame_buff

JPDA 架构研究11 - Agent利用环境指针访问VM(断点操作篇)

引入: 上文我们讲解了Agent如何利用环境指针访问VM(操作局部变量篇),这里主要关注于如何操作断点. 这个在调试期间非常频繁的被使用. 分类7:断点操作 a.SetBreakPoint.设置断点,它会产生一个BreakPoint事件. jvmtiError SetBreakpoint(jvmtiEnv* env,             jmethodID method,             jlocation location) 设置断点指令.从这里可以看出需要传两个参数,一个是被设断

JPDA 架构研究12 - Agent利用环境指针访问VM(观察字段篇)

引入: 上文我们讲解了Agent如何利用环境指针访问VM的操作断点部分,这里我们详细讲解观察字段篇.这也是我们调试期间用的最多的操作.因为我们要不断从Watch Field看他们值的改变. 分类8:观察字段 a.SetFieldAccessWatch.启用对某字段进行访问的观察.每次访问都会创建一个FieldWatch事件. jvmtiError SetFieldAccessWatch(jvmtiEnv* env,             jclass klass,             jf

JPDA 架构研究15 - Agent利用环境指针访问VM(字段访问篇)

引入: 上文中我们已经探讨了Agent如何利用环境指针访问VM(对象级的操作),现在我们 讨论让Agent利用环境指针访问类的字段. 分类10:字段访问 a.GetFieldName.获取某类的某字段名字 jvmtiError GetFieldName(jvmtiEnv* env,             jclass klass,             jfieldID field,             char** name_ptr,             char** signat