Date: Fri, 24 Jan 2003 08:06:59 +0600 From: Max Khon <fjoe@iclub.nsu.ru> To: Bill Huey <billh@gnuppy.monkey.org> Cc: Greg Lewis <glewis@eyesbeyond.com>, freebsd-java@freebsd.org Subject: Re: /usr/local/include/utils.h Message-ID: <20030124080659.A46417@iclub.nsu.ru> In-Reply-To: <20030124015602.GC7116@gnuppy.monkey.org>; from billh@gnuppy.monkey.org on Thu, Jan 23, 2003 at 05:56:02PM -0800 References: <20030123194719.B25462@iclub.nsu.ru> <20030124044501.B76567@misty.eyesbeyond.com> <20030124033021.A38610@iclub.nsu.ru> <20030123214624.GA2919@gnuppy.monkey.org> <20030124062940.A44070@iclub.nsu.ru> <20030124011042.GA3765@gnuppy.monkey.org> <20030124071450.A45276@iclub.nsu.ru> <20030124015602.GC7116@gnuppy.monkey.org>
next in thread | previous in thread | raw e-mail | index | archive | help
--ZPt4rx8FFjLCG7dd Content-Type: text/plain; charset=us-ascii Content-Disposition: inline hi, there! On Thu, Jan 23, 2003 at 05:56:02PM -0800, Bill Huey wrote: > > original e-mail contained an attachement. should I repost to > > mailing list or send it privately? > > Either way is fine, publically would be fine as well as to me > and Greg. It's not very difficult to do, but I've been pretty > unmotivated to do it since I'm trying to move into kernel work. :) hmm, ok, I'll repost it. The file should be placed instead of j2sdk1.3.1/src/solaris/hpi/native_threads/src/threads_bsd.c Basically it contains the following changes: - catchup with latest changes to ucontext (use jmp_buf only) - use pthread_attr_get_np to get thread stack address and its size both should be #ifdef'ed with '#if __FreeBSD_version >= 470101' if we would like to support FreeBSD 4.7-RELEASE and earlier (this will leave some window between 4.7-RELEASE and 4.8-RELEASE where it will not compile but basically it's ok) /fjoe --ZPt4rx8FFjLCG7dd Content-Type: text/plain; charset=us-ascii Content-Disposition: attachment; filename="threads_bsd.c" /* * BSDI $Id: threads_bsd.c,v 1.12 2002/02/07 05:19:54 glewis Exp $ * * from @(#)threads_solaris.c 1.3 98/09/16 * * Copyright 1994-1998 by Sun Microsystems, Inc., * 901 San Antonio Road, Palo Alto, California, 94303, U.S.A. * All rights reserved. * * This software is the confidential and proprietary information * of Sun Microsystems, Inc. ("Confidential Information"). You * shall not disclose such Confidential Information and shall use * it only in accordance with the terms of the license agreement * you entered into with Sun. */ /* * Implementation of HPI that can not be expressed with POSIX threads. */ #include "hpi_impl.h" #include "monitor_md.h" #include "threads_md.h" #include "np.h" #include <pthread.h> #include <pthread_np.h> #if defined(__FreeBSD__) #include <assert.h> #undef pthread_attr_default #undef pthread_mutexattr_default #undef pthread_condattr_default #include "pthread_private.h" #endif #include <string.h> #include <signal.h> #include <sys/signal.h> #include <sys/resource.h> #include <stdlib.h> #include <string.h> #include <errno.h> #include <sys/types.h> #include <sys/sysctl.h> /* * Forward declarations. */ /* Private functions used to implement native threading. --billh */ #ifdef DEBUG_BSD_NATIVE_THREADS void dumpThreadLogStates(pthread_t); void dumpThreadStates(); #endif /* * Suspend a thread. Used to implement java.lang.Thread.suspend(), * which is deprecated. */ int np_suspend(sys_thread_t *tid) { return pthread_suspend_np(tid->sys_thread); } /* * Resume a suspended thread. Used to implement java.lang.Thread.resume(), * which is deprecated. */ int np_continue(sys_thread_t *tid) { return pthread_resume_np(tid->sys_thread); } /* * If there is any initialization is required by the non-POSIX parts. */ void np_initialize_thread(sys_thread_t *tid) { return; } /* * Get the stack start address, and max stack size for the current thread. */ int np_stackinfo(void **addr, long *sizep) { pthread_attr_t attr; size_t size; if ((errno = pthread_attr_init(&attr))) return SYS_ERR; if ((errno = pthread_attr_get_np(pthread_self(), &attr))) goto err; if ((errno = pthread_attr_getstackaddr(&attr, addr))) goto err; if ((errno = pthread_attr_getstacksize(&attr, &size))) goto err; *sizep = size; pthread_attr_destroy(&attr); return SYS_OK; err: pthread_attr_destroy(&attr); return SYS_ERR; } /* * We appear to use this code to suspend threads and get snapshots of * their state from the profiler. If that's the case, then it ought * to suffice to use pthread_suspend_np() and pthread_resume_np(). */ void np_profiler_init(sys_thread_t *tid) { /* We should probably attach the newly allocated pthread's thread structure to the JVM's threading abstraction here, or maybe do nothing () ? --billh */ } int np_profiler_suspend(sys_thread_t *tid) { if (errno = pthread_suspend_np(tid->sys_thread)) return SYS_ERR; return SYS_OK; } int np_profiler_continue(sys_thread_t *tid) { if (errno = pthread_resume_np(tid->sys_thread)) return SYS_ERR; return SYS_OK; } /* * Instead of picking over saved registers, we sum a chunk of the stack. */ bool_t np_profiler_thread_is_running(sys_thread_t *tid) { /* This function didn't show up any where in a grep of all the sources, nor do any of the other non-posix implementations (Linux, Solaris) have a function body more complicated than the single statement "return TRUE;". Do this for the FreeBSD implementation too, since this is a silly function anyways. --billh */ return TRUE; } int np_initialize() { return SYS_OK; } static void record_thread_regs() { struct pthread *self = pthread_self(); sys_thread_t *tid = ThreadQueue; int i; for (i = 0; i < ActiveThreadCount && tid != NULL; i++, tid = tid->next) { struct pthread *thread = tid->sys_thread; if (thread == 0) { /* * thread is still in the process of being initalized. * So GC should not care about this thread. Just * set its sp to 0, and this will force GC to ignore it. */ tid->sp = 0; continue; } tid->sp = thread->stack; /* Potential race here if the stack isn't setup before GC. --billh */ /* * The thread that calls this function will alway be the JVM GC thread, * so skip over it in the list of threads. */ if (thread != self && (thread->flags & PTHREAD_FLAGS_PRIVATE) == 0) { register_t *regbase; #ifdef DEBUG_BSD_NATIVE_THREADS /* * Got search candidate.. */ if (thread->state != PS_SUSPENDED) dumpThreadLogStates(thread); #endif regbase = (register_t*) &thread->ctx.jb[0]; tid->regs[0] = regbase[6]; /* eax */ tid->regs[1] = 0; /* ecx (missing) */ tid->regs[2] = 0; /* edx (missing) */ tid->regs[3] = regbase[1]; /* ebx */ tid->regs[4] = regbase[3]; /* ebp */ tid->regs[5] = regbase[4]; /* esi */ tid->regs[6] = regbase[5]; /* edi */ #ifdef DEBUG_BSD_NATIVE_THREADS dumpThreadStates(); #endif } } #ifdef DEBUG_BSD_NATIVE_THREADS fprintf(stderr, "\n\n\nCalling GC thread\n\n\n"); fflush(stderr); #endif } /* * Suspend all other threads, and record their contexts (register * set or stack pointer) into the sys_thread structure, so that a * garbage collect can be run. */ int np_single(void) { sysAssert(SYS_QUEUE_LOCKED(sysThreadSelf())); pthread_suspend_all_np(); record_thread_regs(); return SYS_OK; } /* * Continue threads suspended earlier. But clear their context * recorded in sys_thread structure first. */ void np_multi(void) { sysAssert(SYS_QUEUE_LOCKED(sysThreadSelf())); pthread_resume_all_np(); } #ifdef DEBUG_BSD_NATIVE_THREADS /* pthreads hackery begins --billh */ char SignalList [][16] = { "PS_RUNNING", "PS_SIGTHREAD", "PS_MUTEX_WAIT", "PS_COND_WAIT", "PS_FDLR_WAIT", "PS_FDLW_WAIT", "PS_FDR_WAIT", "PS_FDW_WAIT", "PS_FILE_WAIT", "PS_POLL_WAIT", "PS_SELECT_WAIT", "PS_SLEEP_WAIT", "PS_WAIT_WAIT", "PS_SIGSUSPEND", "PS_SIGWAIT", "PS_SPINBLOCK", "PS_JOIN", "PS_SUSPENDED", "PS_DEAD", "PS_DEADLOCK", "PS_STATE_MAX", "boundless" }; char *getThreadStateString(enum pthread_state threadState) { if (threadState < SATAN) return SignalList[threadState]; else return SignalList[SATAN-1]; } void dumpThreadStates() { int threadCount = 0; struct pthread *thread; struct pthread *self = pthread_self(); _thread_kern_sig_defer(); TAILQ_FOREACH(thread, &_thread_list, tle) { if (thread != self) { /* special case this --billh */ printf("\tthread %d\t%s\n", threadCount, getThreadStateString(thread->state)); if (thread->state != PS_SUSPENDED) dumpThreadLogStates(thread); } else { printf("\tgc thread %d\t%s\n", threadCount, getThreadStateString(thread->state)) } ++threadCount; } _thread_kern_sig_undefer(); printf("\n"); } /* [A snippet from Dan Eichen's email on the subject] You can also look at src/gnu/usr.bin/binutils/gdb/freebsd-uthread.c. It knows how to iterate through all the threads and pull out (and even set) thread contexts. All the threads are stored in a TAILQ called _thread_list. There is one thread that is private to the threads library (uthread_gc.c) and is marked with thread->flags & PTHREAD_FLAGS_PRIVATE. I don't think you want to include that thread in whatever you are doing, but that's up to you to decide. [end] 1) Write the code to extract the x86's registers from the jmp_buf and ucontext_t. --billh */ void dumpThreadLogStates(pthread_t thread) { int i; for(i=0; i < STATE_LOG_SIZE; ++i) { printf("\t\t%s, %s\n", getThreadStateString (thread->state_log[i].state), getSuspendStateString(thread->suspended) ); printf("\t\tState change fname = %s, lineno = %d, state = %d\n", thread->state_log[i].fname, thread->state_log[i].lineno, thread->state_log[i].state); if (thread->state_log[i].state == PS_SUSPENDED) { printf("\t\t***\n"); return; } } printf("\t\t***XXX\n"); } #endif --ZPt4rx8FFjLCG7dd-- To Unsubscribe: send mail to majordomo@FreeBSD.org with "unsubscribe freebsd-java" in the body of the message
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?20030124080659.A46417>