1 // SPDX-License-Identifier: GPL-2.0-only 1 2 /* 3 * Generic polynomial calculation using intege 4 * 5 * Copyright (C) 2020 BAIKAL ELECTRONICS, JSC 6 * 7 * Authors: 8 * Maxim Kaurkin <maxim.kaurkin@baikalelectr 9 * Serge Semin <Sergey.Semin@baikalelectroni 10 * 11 */ 12 13 #include <linux/kernel.h> 14 #include <linux/module.h> 15 #include <linux/polynomial.h> 16 17 /* 18 * Originally this was part of drivers/hwmon/b 19 * There the following conversion is used and 20 * 21 * The original translation formulae of the te 22 * to PVT data and vice-versa are following: 23 * 24 * N = 1.8322e-8*(T^4) + 2.343e-5*(T^3) + 8.70 25 * 1.7204e2 26 * T = -1.6743e-11*(N^4) + 8.1542e-8*(N^3) + - 27 * 3.1020e-1*(N^1) - 4.838e1 28 * 29 * where T = [-48.380, 147.438]C and N = [0, 1 30 * 31 * They must be accordingly altered to be suit 32 * The technique is called 'factor redistribut 33 * multiplications and divisions are made so t 34 * within the integer numbers limit. In additi 35 * formulae to accept millidegrees of Celsius. 36 * the alterations: 37 * 38 * N = (18322e-20*(T^4) + 2343e-13*(T^3) + 870 39 * 17204e2) / 1e4 40 * T = -16743e-12*(D^4) + 81542e-9*(D^3) - 182 41 * 48380 42 * where T = [-48380, 147438] mC and N = [0, 1 43 * 44 * static const struct polynomial poly_temp_to 45 * .total_divider = 10000, 46 * .terms = { 47 * {4, 18322, 10000, 10000}, 48 * {3, 2343, 10000, 10}, 49 * {2, 87018, 10000, 10}, 50 * {1, 39269, 1000, 1}, 51 * {0, 1720400, 1, 1} 52 * } 53 * }; 54 * 55 * static const struct polynomial poly_N_to_te 56 * .total_divider = 1, 57 * .terms = { 58 * {4, -16743, 1000, 1}, 59 * {3, 81542, 1000, 1}, 60 * {2, -182010, 1000, 1}, 61 * {1, 310200, 1000, 1}, 62 * {0, -48380, 1, 1} 63 * } 64 * }; 65 */ 66 67 /** 68 * polynomial_calc - calculate a polynomial us 69 * 70 * @poly: pointer to the descriptor of the pol 71 * @data: input value of the polynimal 72 * 73 * Calculate the result of a polynomial using 74 * this to work without too much loss of preci 75 * be altered. This is called factor redistrib 76 * 77 * Returns the result of the polynomial calcul 78 */ 79 long polynomial_calc(const struct polynomial * 80 { 81 const struct polynomial_term *term = p 82 long total_divider = poly->total_divid 83 long tmp, ret = 0; 84 int deg; 85 86 /* 87 * Here is the polynomial calculation 88 * redistributed terms calculations. I 89 * We walk over each degree term up to 90 * the redistributed multiplication of 91 * divider (as for the rationale fract 92 * power and the rational fraction div 93 * this is collected in a total sum va 94 * normalized by the total divider bef 95 */ 96 do { 97 tmp = term->coef; 98 for (deg = 0; deg < term->deg; 99 tmp = mult_frac(tmp, d 100 ret += tmp / term->divider_lef 101 } while ((term++)->deg); 102 103 return ret / total_divider; 104 } 105 EXPORT_SYMBOL_GPL(polynomial_calc); 106 107 MODULE_DESCRIPTION("Generic polynomial calcula 108 MODULE_LICENSE("GPL"); 109
Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.