xref: /netbsd-src/lib/libc/stdlib/div.c (revision 9e66e6d75e9910b3de5f4ef031995955f69a7dd1)
1*9e66e6d7Sabs /*	$NetBSD: div.c,v 1.8 2012/06/25 22:32:45 abs Exp $	*/
26dda330eSthorpej 
361f28255Scgd /*
45f8a6c06Sperry  * Copyright (c) 1990, 1993
55f8a6c06Sperry  *	The Regents of the University of California.  All rights reserved.
661f28255Scgd  *
761f28255Scgd  * This code is derived from software contributed to Berkeley by
861f28255Scgd  * Chris Torek.
961f28255Scgd  *
1061f28255Scgd  * Redistribution and use in source and binary forms, with or without
1161f28255Scgd  * modification, are permitted provided that the following conditions
1261f28255Scgd  * are met:
1361f28255Scgd  * 1. Redistributions of source code must retain the above copyright
1461f28255Scgd  *    notice, this list of conditions and the following disclaimer.
1561f28255Scgd  * 2. Redistributions in binary form must reproduce the above copyright
1661f28255Scgd  *    notice, this list of conditions and the following disclaimer in the
1761f28255Scgd  *    documentation and/or other materials provided with the distribution.
18eb7c1594Sagc  * 3. Neither the name of the University nor the names of its contributors
1961f28255Scgd  *    may be used to endorse or promote products derived from this software
2061f28255Scgd  *    without specific prior written permission.
2161f28255Scgd  *
2261f28255Scgd  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
2361f28255Scgd  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2461f28255Scgd  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2561f28255Scgd  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2661f28255Scgd  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2761f28255Scgd  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2861f28255Scgd  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2961f28255Scgd  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
3061f28255Scgd  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
3161f28255Scgd  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3261f28255Scgd  * SUCH DAMAGE.
3361f28255Scgd  */
3461f28255Scgd 
35bd906777Schristos #include <sys/cdefs.h>
3661f28255Scgd #if defined(LIBC_SCCS) && !defined(lint)
376dda330eSthorpej #if 0
385f8a6c06Sperry static char sccsid[] = "@(#)div.c	8.1 (Berkeley) 6/4/93";
396dda330eSthorpej #else
40*9e66e6d7Sabs __RCSID("$NetBSD: div.c,v 1.8 2012/06/25 22:32:45 abs Exp $");
416dda330eSthorpej #endif
4261f28255Scgd #endif /* LIBC_SCCS and not lint */
4361f28255Scgd 
4461f28255Scgd #include <stdlib.h>		/* div_t */
4561f28255Scgd 
4661f28255Scgd div_t
div(int num,int denom)47*9e66e6d7Sabs div(int num, int denom)
4861f28255Scgd {
4961f28255Scgd 	div_t r;
5061f28255Scgd 
5161f28255Scgd 	r.quot = num / denom;
5261f28255Scgd 	r.rem = num % denom;
5361f28255Scgd 	/*
5461f28255Scgd 	 * The ANSI standard says that |r.quot| <= |n/d|, where
5561f28255Scgd 	 * n/d is to be computed in infinite precision.  In other
5661f28255Scgd 	 * words, we should always truncate the quotient towards
5761f28255Scgd 	 * 0, never -infinity.
5861f28255Scgd 	 *
5961f28255Scgd 	 * Machine division and remainer may work either way when
6061f28255Scgd 	 * one or both of n or d is negative.  If only one is
6161f28255Scgd 	 * negative and r.quot has been truncated towards -inf,
6261f28255Scgd 	 * r.rem will have the same sign as denom and the opposite
6361f28255Scgd 	 * sign of num; if both are negative and r.quot has been
6461f28255Scgd 	 * truncated towards -inf, r.rem will be positive (will
6561f28255Scgd 	 * have the opposite sign of num).  These are considered
6661f28255Scgd 	 * `wrong'.
6761f28255Scgd 	 *
6861f28255Scgd 	 * If both are num and denom are positive, r will always
6961f28255Scgd 	 * be positive.
7061f28255Scgd 	 *
7161f28255Scgd 	 * This all boils down to:
7261f28255Scgd 	 *	if num >= 0, but r.rem < 0, we got the wrong answer.
7361f28255Scgd 	 * In that case, to get the right answer, add 1 to r.quot and
7461f28255Scgd 	 * subtract denom from r.rem.
7561f28255Scgd 	 */
7661f28255Scgd 	if (num >= 0 && r.rem < 0) {
7761f28255Scgd 		r.quot++;
7861f28255Scgd 		r.rem -= denom;
7961f28255Scgd 	}
8061f28255Scgd 	return (r);
8161f28255Scgd }
82