]> git.ipfire.org Git - thirdparty/u-boot.git/commitdiff
lib/bcd: optimize _bin2bcd() for improved performance
authorKuan-Wei Chiu <visitorckw@gmail.com>
Thu, 11 Dec 2025 18:31:30 +0000 (18:31 +0000)
committerTom Rini <trini@konsulko.com>
Fri, 2 Jan 2026 21:51:54 +0000 (15:51 -0600)
[ Upstream commit cbf164cd44e06c78938b4a4a4479d3541779c319 ]

The original _bin2bcd() function used / 10 and % 10 operations for
conversion.  Although GCC optimizes these operations and does not generate
division or modulus instructions, the new implementation reduces the
number of mov instructions in the generated code for both x86-64 and ARM
architectures.

This optimization calculates the tens digit using (val * 103) >> 10, which
is accurate for values of 'val' in the range [0, 178].  Given that the
valid input range is [0, 99], this method ensures correctness while
simplifying the generated code.

Link: https://lkml.kernel.org/r/20240812170229.229380-1-visitorckw@gmail.com
Signed-off-by: Kuan-Wei Chiu <visitorckw@gmail.com>
Cc: Ching-Chun Huang (Jim) <jserv@ccns.ncku.edu.tw>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
[visitorckw@gmail.com: Adapt to bin2bcd() in include/bcd.h]
Signed-off-by: Kuan-Wei Chiu <visitorckw@gmail.com>
include/bcd.h

index 9ecd328284e9de20a293a96dff9bc655c6d529ea..289810b99b71745efb2bf55653ede939765c8b92 100644 (file)
@@ -17,7 +17,9 @@ static inline unsigned int bcd2bin(unsigned int val)
 
 static inline unsigned int bin2bcd(unsigned int val)
 {
-       return (((val / 10) << 4) | (val % 10));
+       const unsigned int t = (val * 103) >> 10;
+
+       return (t << 4) | (val - t * 10);
 }
 
 #endif /* _BCD_H */