1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
| #include <stdio.h>
#include <stdlib.h>
#include <limits.h>
enum AddErrors {
ADD_OK,
ADD_ERR_ARG_OVERFLOW,
ADD_ERR_OVERFLOW_MAX,
ADD_ERR_OVERFLOW_MIN,
ADD_NB
};
/**
* Additionne deux valeurs codées sur 16bits et signale les débobordements.
*
* Si les valeurs passees en arguments ne peuvent etre representees par un
* codage sur 16 bits, la fonction revoie 0 et err vaut ADD_ERR_ARG_OVERFLOW.
* Si un debordement intervient, la fonction renvoie 0 et err vaut
* ADD_ERR_OVERFLOW_MAX (debordement par le haut) ou ADD_ERR_OVERFLOW_MIN
* (debordement par le bas).
*/
int add_16bits_numbers(int nb1, int nb2, enum AddErrors *err)
{
/* -tc- valeur de retour */
int rv = 0;
/* -tc- teste si un entier negatif est repesente selon un complement a deux */
int is_two_complement = (INT_MAX == - (INT_MIN + 1));
const int MAX_SIGNED_16BITS = 32767;
const int MIN_SIGNED_16BITS = - (MAX_SIGNED_16BITS + is_two_complement);
enum AddErrors error = ADD_OK;
/* -tc- si les valeurs passees en argument ne sont pas comprises entre
MAX_SIGNED_16BITS et MIN_SIGNED_16BITS, la fonction renvoie 0 et une
erreur ADD_ERR_ARG_OVERFLOW */
if ((nb1 <= MAX_SIGNED_16BITS && nb1 >= MIN_SIGNED_16BITS) &&
(nb1 <= MAX_SIGNED_16BITS && nb2 >= MIN_SIGNED_16BITS))
{
/* -tc- il ne peut y avoir debordement que si les valeurs sont les
deux positives ou les deux negatives */
if (nb1 > 0 && nb2 > 0 && nb1 > MAX_SIGNED_16BITS - nb2)
{
error = ADD_ERR_OVERFLOW_MAX;
}
else if (nb1 < 0 && nb2 < 0 && nb1 < MIN_SIGNED_16BITS - nb2)
{
error = ADD_ERR_OVERFLOW_MIN;
}
else /* -tc- il n'y a pas de debordement */
{
rv = nb1 + nb2;
}
}
else
{
error = ADD_ERR_ARG_OVERFLOW;
}
if (err != NULL)
{
*err = error;
}
return rv;
}
int main(void)
{
enum AddErrors err = ADD_OK;
short nb1 = 32767;
short nb2 = 1;
int res;
res = add_16bits_numbers(nb1, nb2, &err);
printf("res = %d\nerr = %d\n", res, err);
return 0;
} |