Obtaining the thread ID for Java threads in Linux

12,399

Solution 1

In the end, I found the JNI way to be the best one to solve my problem. As a reference, I post the code and build instructions for it (based on the example at Wikipedia):

Java class responsible to interface to the C code (GetThreadID.java):

public class GetThreadID {
    public static native int get_tid();

    static {
        System.loadLibrary("GetThreadID");
    }
}

C file responsible to obtain the thread ID (GetThread.c):

#include <jni.h>
#include <syscall.h>
#include "GetThreadID.h"

JNIEXPORT jint JNICALL
Java_GetThreadID_get_1tid(JNIEnv *env, jobject obj) {
    jint tid = syscall(__NR_gettid);
    return tid;
}

An example of how to use GetThreadID class:

class Main {
    public static void main(String[] args) {
        int tid = GetThreadID.get_tid();
        System.out.println("TID=" + tid);
    }
}

And finally, the build instructions (javah automatically generates GetThreadID.h):

JAVA_HOME=$(readlink -f /usr/bin/javac | sed "s:bin/javac::")
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:.
javac GetThreadID.java
javah GetThreadID

gcc -I${JAVA_HOME}/include -fPIC -shared GetThreadID.c -o libGetThreadID.so
javac Main.java
java Main

Solution 2

This blog post provides a method of mapping from java thread ids to LWP ids.

The gist of it seems to be that the NLWP id maps to the java thread id.

Share:
12,399
betabandido
Author by

betabandido

PhD in computer architecture with experience in operating systems, runtimes, micro-architecture simulation, performance analysis, optimization, workload development and heterogeneous systems. Currently working as a software engineer in multiple projects involving different technologies and practices such as microservices, cloud computing, infrastructure-as-code and devops.

Updated on July 19, 2022

Comments

  • betabandido
    betabandido almost 2 years

    I have a Java application where some threads are created (via new Thread()). Using ps I can see they have different thread IDs (LWP column) and I would like to obtain those IDs from within the Java application.

    In most of the posts related to this topic that I have found (e.g., this one), the solution is to use ManagementFactory.getRuntimeMXBean().getName().

    Using that method, however, gives me the PID of the main thread (even if I call it from one of the threads), so it is not really solving my problem.

    Is there any way to obtain the thread ID for every single Thread created by an application?

    Would it be possible to use JNI to accomplish it? If somehow I could interface to a C function where I could call syscall(__NR_gettid), that could solve my problem. I really do not care about portability, so I am totally okay with a solution that would only work for a Linux machine.

    UPDATE: I have actually solved my problem by using JNI. Details are explained in my answer. Thank you all for your suggestions/comments.

  • Niklas B.
    Niklas B. almost 12 years
    I think that blog post talks about manually mapping the threads. OP wants to do it automatically from within Java
  • Sean Reilly
    Sean Reilly almost 12 years
    @NiklasB: I know. My point was that it probably can be done automatically if the OP is willing to map the thread ids to NLWP id instead of LWP id.
  • betabandido
    betabandido almost 12 years
    @SeanReilly This definitely looks promising. As you are saying, it could be automated. I will explore this way, as well as the JNI (which would be more direct, but I do not know yet whether it will work). Thanks!
  • Niklas B.
    Niklas B. almost 12 years
    Very nice, thanks for the follow-up. I suspected it wouldn't be too much work :) I think you should accept your own answer here, as it's definitely the best answer.
  • gustavohenke
    gustavohenke over 7 years
    Please don't post link only answers.
  • Sean Reilly
    Sean Reilly over 7 years
    @gustavohenke: Thanks for the very timely comment. ;-)
  • Artanis Zeratul
    Artanis Zeratul over 5 years
    I think the gcc instruction here should be gcc -I${JAVA_HOME}/include -I${JAVA_HOME}/include/linux -fPIC -shared GetThread.c -o libGetThreadID.so now a days :D!
  • Artanis Zeratul
    Artanis Zeratul over 5 years
    Apparently this won't work if you will omit JAVA_HOME part. Specially if you have set the JAVA_HOME already. But if you follow the instructions plus my modifications this works.