4470 overly aggressive D integer narrowing breaks 32-bit ustack helpers

illumos/illumos-gate@2e0552679e
This commit is contained in:
Mark Johnston 2014-06-03 01:55:27 +00:00
parent ced1f492b7
commit ba911031dd
2 changed files with 49 additions and 1 deletions

View File

@ -0,0 +1,48 @@
#
# CDDL HEADER START
#
# The contents of this file are subject to the terms of the
# Common Development and Distribution License (the "License").
# You may not use this file except in compliance with the License.
#
# You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
# or http://www.opensolaris.org/os/licensing.
# See the License for the specific language governing permissions
# and limitations under the License.
#
# When distributing Covered Code, include this CDDL HEADER in each
# file and include the License file at usr/src/OPENSOLARIS.LICENSE.
# If applicable, add the following below this CDDL HEADER, with the
# fields enclosed by brackets "[]" replaced with your own identifying
# information: Portions Copyright [yyyy] [name of copyright owner]
#
# CDDL HEADER END
#
#
# Copyright (c) 2012, Joyent, Inc. All rights reserved.
#
#
# Somewhat surprisingly, DTrace very much relies on the fact that when 32-bit
# is forced, pointers are not, in fact, narrowed to 32-bits. In particular, if
# this is not so (that is, if pointers are narrowed to their seemingly correct
# 32-bit width), helpers attached to 32-bit programs will fail to operate:
# they will erroneously zero the high 32-bits of the return values of 64-bit
# kernel pointers as returned by copyin(), alloca(), etc. This test asserts
# this implicit behavior -- and this comment regrettably serves as this
# behavior's only documentation.
#
doit()
{
/usr/sbin/dtrace $1 -n BEGIN'{trace(sizeof (long))}' \
-n 'BEGIN{*(int *)alloca(4) = 21506; exit(0)}' \
-n 'ERROR{exit(1)}'
if [ "$?" -ne 0 ]; then
exit $?
fi
}
doit
doit -32

View File

@ -476,7 +476,7 @@ dt_cg_typecast(const dt_node_t *src, const dt_node_t *dst,
if (!dt_node_is_scalar(dst))
return; /* not a scalar */
if (dstsize == srcsize &&
((src->dn_flags ^ dst->dn_flags) & DT_NF_SIGNED) != 0)
((src->dn_flags ^ dst->dn_flags) & DT_NF_SIGNED) == 0)
return; /* not narrowing or changing signed-ness */
if (dstsize > srcsize && (src->dn_flags & DT_NF_SIGNED) == 0)
return; /* nothing to do in this case */