/* * Do a strnlen, return length of string *with* final '\0'. * 'count' is the user-supplied count, while 'max' is the * address space maximum. * * Return 0 for exceptions (which includes hitting the address * space maximum), or 'count+1' if hitting the user-supplied * maximum count. * * NOTE! We can sometimes overshoot the user-supplied maximum * if it fits in a aligned 'long'. The caller needs to check * the return value against "> max".
*/ static __always_inline long do_strnlen_user(constchar __user *src, unsignedlong count, unsignedlong max)
{ conststruct word_at_a_time constants = WORD_AT_A_TIME_CONSTANTS; unsignedlong align, res = 0; unsignedlong c;
/* * Do everything aligned. But that means that we * need to also expand the maximum..
*/
align = (sizeof(unsignedlong) - 1) & (unsignedlong)src;
src -= align;
max += align;
unsafe_get_user(c, (unsignedlong __user *)src, efault);
c |= aligned_byte_mask(align);
for (;;) { unsignedlong data; if (has_zero(c, &data, &constants)) {
data = prep_zero_mask(c, data, &constants);
data = create_zero_mask(data); return res + find_zero(data) + 1 - align;
}
res += sizeof(unsignedlong); /* We already handled 'unsigned long' bytes. Did we do it all ? */ if (unlikely(max <= sizeof(unsignedlong))) break;
max -= sizeof(unsignedlong);
unsafe_get_user(c, (unsignedlong __user *)(src+res), efault);
}
res -= align;
/* * Uhhuh. We hit 'max'. But was that the user-specified maximum * too? If so, return the marker for "too long".
*/ if (res >= count) return count+1;
/* * Nope: we hit the address space limit, and we still had more * characters the caller would have wanted. That's 0.
*/
efault: return 0;
}
/** * strnlen_user: - Get the size of a user string INCLUDING final NUL. * @str: The string to measure. * @count: Maximum count (including NUL character) * * Context: User context only. This function may sleep if pagefaults are * enabled. * * Get the size of a NUL-terminated string in user space. * * Returns the size of the string INCLUDING the terminating NUL. * If the string is too long, returns a number larger than @count. User * has to check the return value against "> count". * On exception (or invalid count), returns 0. * * NOTE! You should basically never use this function. There is * almost never any valid case for using the length of a user space * string, since the string can be changed at any time by other * threads. Use "strncpy_from_user()" instead to get a stable copy * of the string.
*/ long strnlen_user(constchar __user *str, long count)
{ unsignedlong max_addr, src_addr;
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.