2N/A/*
2N/A * CDDL HEADER START
2N/A *
2N/A * The contents of this file are subject to the terms of the
2N/A * Common Development and Distribution License (the "License").
2N/A * You may not use this file except in compliance with the License.
2N/A *
2N/A * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
2N/A * or http://www.opensolaris.org/os/licensing.
2N/A * See the License for the specific language governing permissions
2N/A * and limitations under the License.
2N/A *
2N/A * When distributing Covered Code, include this CDDL HEADER in each
2N/A * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
2N/A * If applicable, add the following below this CDDL HEADER, with the
2N/A * fields enclosed by brackets "[]" replaced with your own identifying
2N/A * information: Portions Copyright [yyyy] [name of copyright owner]
2N/A *
2N/A * CDDL HEADER END
2N/A */
2N/A/*
2N/A * Copyright 2004 Sun Microsystems, Inc. All rights reserved.
2N/A * Use is subject to license terms.
2N/A */
2N/A
2N/A .file "tls_get_addr.s"
2N/A
2N/A/*
2N/A * To make thread-local storage accesses as fast as possible, we
2N/A * hand-craft the __tls_get_addr() function below, from this C code:
2N/A * void *
2N/A * __tls_get_addr(TLS_index *tls_index)
2N/A * {
2N/A * ulwp_t *self = curthread;
2N/A * tls_t *tlsent = self->ul_tlsent;
2N/A * ulong_t moduleid;
2N/A * caddr_t base;
2N/A *
2N/A * if ((moduleid = tls_index->ti_moduleid) < self->ul_ntlsent &&
2N/A * (base = tlsent[moduleid].tls_data) != NULL)
2N/A * return (base + tls_index->ti_tlsoffset);
2N/A *
2N/A * return (slow_tls_get_addr(tls_index));
2N/A * }
2N/A *
2N/A * ___tls_get_addr() is identical to __tls_get_addr() except that it
2N/A * assumes its argument is passed in %eax rather than on the stack.
2N/A */
2N/A
2N/A#include "SYS.h"
2N/A#include <assym.h>
2N/A
2N/A ENTRY_NP(__tls_get_addr)
2N/A movl 4(%esp), %eax
2N/A ALTENTRY(___tls_get_addr)
2N/A movl %gs:UL_TLSENT, %edx
2N/A movl TI_MODULEID (%eax), %ecx
2N/A cmpl %gs:UL_NTLSENT, %ecx
2N/A jae 1f
2N/A movl TLS_DATA (%edx,%ecx,SIZEOF_TLS_T), %edx
2N/A testl %edx, %edx
2N/A je 1f
2N/A addl TI_TLSOFFSET (%eax), %edx
2N/A movl %edx, %eax
2N/A ret
2N/A1:
2N/A pushl %ebp
2N/A movl %esp, %ebp
2N/A pushl %eax
2N/A call slow_tls_get_addr
2N/A addl $4, %esp
2N/A leave
2N/A ret
2N/A SET_SIZE(___tls_get_addr)
2N/A SET_SIZE(__tls_get_addr)