Received: by 2002:a05:6358:11c7:b0:104:8066:f915 with SMTP id i7csp5770617rwl; Tue, 11 Apr 2023 09:42:58 -0700 (PDT) X-Google-Smtp-Source: AKy350ZMyR7nyHn5ppUtWXbOb0SXE7OQ1GFeBS5nYGwP4ybvgwFZg4XudcJYvuD6ketRU09qHLMr X-Received: by 2002:a17:906:32d1:b0:8b1:fc1a:7d21 with SMTP id k17-20020a17090632d100b008b1fc1a7d21mr12681553ejk.5.1681231377563; Tue, 11 Apr 2023 09:42:57 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1681231377; cv=none; d=google.com; s=arc-20160816; b=QxMKTheyxeXGcY0qwJ14iI5zxQ1C//qZdbARFLjdZABJF2fcjD2TiN/bdyTcMk72CO rimv8p0Dl8eFkS655+vjYyJAoLXTEpT1wizoMDNZS9wfWNjjqnjJOsRZZCK12iTNxi59 ppdJzaLNHt0xdwcT4LdIqpz89spfzkJDFGE7OJ1w1BZgmQMwJQcAoHZzjVnlcutLVzlx rHQ9ADHpKrajYiqzWYnv7YyUKxs1Kt2DrurH1IDeVf4YbPzDn53nlovWphJ1Yr23gKJB 3Sc3nUfYGg9rulRSWok+Lav9CkthbMEp+BRppx0PBKuqkX5OxWvaoGzNmCnm/sxUB8on t6Cw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:in-reply-to:content-disposition:mime-version :references:message-id:subject:cc:to:from:date; bh=1iML3d2c9pIwUCCChFeMNhVdl54gQcQ7LLUpS91fepQ=; b=F3d3XBoVkukq+jBJqAsT5n4appkHoN34VtunXb3n9+3xhI92WIXTZ8RiNowhulOGkJ OgFlqDSZqq3XYJSuWfZo0DTxBWYbTFxyQzu+oQ1b1IMk9lpKsNEumxrwXaEhTHmDkiTh BgRcWRLdjJy5FSCphlWLPRd6kS8zQkfwMh4Y3yeAH1BcURUwMmqJq0gf6tVClhvoPFrg 4RjChwBWO61glhlS40Y8l66TeMjDozo5NHqXNkVco5ry1TQ045gDmY7mriflsbJD4DHA 8ildgFCBwWx491ZjJMSOWgfCfEoW2DYPtgPm7Kwbdi7THIq4a2PL8v2/NxDDjU1ya4do WBWQ== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id j3-20020a1709064b4300b0094a780acf82si5443369ejv.211.2023.04.11.09.42.32; Tue, 11 Apr 2023 09:42:57 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S229831AbjDKQ0N (ORCPT + 99 others); Tue, 11 Apr 2023 12:26:13 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60476 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229807AbjDKQ0L (ORCPT ); Tue, 11 Apr 2023 12:26:11 -0400 Received: from metis.ext.pengutronix.de (metis.ext.pengutronix.de [IPv6:2001:67c:670:201:290:27ff:fe1d:cc33]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 0379D30CA for ; Tue, 11 Apr 2023 09:26:08 -0700 (PDT) Received: from drehscheibe.grey.stw.pengutronix.de ([2a0a:edc0:0:c01:1d::a2]) by metis.ext.pengutronix.de with esmtps (TLS1.3:ECDHE_RSA_AES_256_GCM_SHA384:256) (Exim 4.92) (envelope-from ) id 1pmGok-0002ye-Nd; Tue, 11 Apr 2023 18:26:02 +0200 Received: from [2a0a:edc0:0:900:1d::77] (helo=ptz.office.stw.pengutronix.de) by drehscheibe.grey.stw.pengutronix.de with esmtp (Exim 4.94.2) (envelope-from ) id 1pmGoi-00AXqG-Tn; Tue, 11 Apr 2023 18:26:00 +0200 Received: from ukl by ptz.office.stw.pengutronix.de with local (Exim 4.94.2) (envelope-from ) id 1pmGoh-00CLBF-Na; Tue, 11 Apr 2023 18:25:59 +0200 Date: Tue, 11 Apr 2023 18:25:54 +0200 From: Uwe =?utf-8?Q?Kleine-K=C3=B6nig?= To: Conor Dooley Cc: Thierry Reding , Daire McNamara , linux-kernel@vger.kernel.org, linux-pwm@vger.kernel.org, linux-riscv@lists.infradead.org Subject: Re: [PATCH v16 1/2] pwm: add microchip soft ip corePWM driver Message-ID: <20230411162554.4tl2z2tcbfg5hb7e@pengutronix.de> References: <20230411-wizard-cautious-3c048db6b4d2@wendy> <20230411-bronzing-crust-d302408a1259@wendy> <20230411105547.ypkktubgfx4jfen3@pengutronix.de> <20230411-ligament-wagon-6c8cacb966e8@wendy> MIME-Version: 1.0 Content-Type: multipart/signed; micalg=pgp-sha512; protocol="application/pgp-signature"; boundary="yrwz5jovccfyp3yp" Content-Disposition: inline In-Reply-To: <20230411-ligament-wagon-6c8cacb966e8@wendy> X-SA-Exim-Connect-IP: 2a0a:edc0:0:c01:1d::a2 X-SA-Exim-Mail-From: ukl@pengutronix.de X-SA-Exim-Scanned: No (on metis.ext.pengutronix.de); SAEximRunCond expanded to false X-PTX-Original-Recipient: linux-kernel@vger.kernel.org X-Spam-Status: No, score=-2.3 required=5.0 tests=RCVD_IN_DNSWL_MED, SPF_HELO_NONE,SPF_PASS autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org --yrwz5jovccfyp3yp Content-Type: text/plain; charset=utf-8 Content-Disposition: inline Content-Transfer-Encoding: quoted-printable Hello Conor, On Tue, Apr 11, 2023 at 02:56:15PM +0100, Conor Dooley wrote: > On Tue, Apr 11, 2023 at 12:55:47PM +0200, Uwe Kleine-K=C3=B6nig wrote: > > On Tue, Apr 11, 2023 at 09:56:34AM +0100, Conor Dooley wrote: > > > Add a driver that supports the Microchip FPGA "soft" PWM IP core. > > >=20 > > > Signed-off-by: Conor Dooley > > > --- > > > drivers/pwm/Kconfig | 10 + > > > drivers/pwm/Makefile | 1 + > > > drivers/pwm/pwm-microchip-core.c | 509 +++++++++++++++++++++++++++++= ++ > > > 3 files changed, 520 insertions(+) > > > create mode 100644 drivers/pwm/pwm-microchip-core.c > > >=20 > > > diff --git a/drivers/pwm/Kconfig b/drivers/pwm/Kconfig > > > index dae023d783a2..f42756a014ed 100644 > > > --- a/drivers/pwm/Kconfig > > > +++ b/drivers/pwm/Kconfig > > > @@ -393,6 +393,16 @@ config PWM_MEDIATEK > > > To compile this driver as a module, choose M here: the module > > > will be called pwm-mediatek. > > > =20 > > > +config PWM_MICROCHIP_CORE > > > + tristate "Microchip corePWM PWM support" > > > + depends on SOC_MICROCHIP_POLARFIRE || COMPILE_TEST > > > + depends on HAS_IOMEM && OF > > > + help > > > + PWM driver for Microchip FPGA soft IP core. > > > + > > > + To compile this driver as a module, choose M here: the module > > > + will be called pwm-microchip-core. > > > + > > > config PWM_MXS > > > tristate "Freescale MXS PWM support" > > > depends on ARCH_MXS || COMPILE_TEST > > > diff --git a/drivers/pwm/Makefile b/drivers/pwm/Makefile > > > index 7bf1a29f02b8..a65625359ece 100644 > > > --- a/drivers/pwm/Makefile > > > +++ b/drivers/pwm/Makefile > > > @@ -34,6 +34,7 @@ obj-$(CONFIG_PWM_LPSS_PCI) +=3D pwm-lpss-pci.o > > > obj-$(CONFIG_PWM_LPSS_PLATFORM) +=3D pwm-lpss-platform.o > > > obj-$(CONFIG_PWM_MESON) +=3D pwm-meson.o > > > obj-$(CONFIG_PWM_MEDIATEK) +=3D pwm-mediatek.o > > > +obj-$(CONFIG_PWM_MICROCHIP_CORE) +=3D pwm-microchip-core.o > > > obj-$(CONFIG_PWM_MTK_DISP) +=3D pwm-mtk-disp.o > > > obj-$(CONFIG_PWM_MXS) +=3D pwm-mxs.o > > > obj-$(CONFIG_PWM_NTXEC) +=3D pwm-ntxec.o > > > diff --git a/drivers/pwm/pwm-microchip-core.c b/drivers/pwm/pwm-micro= chip-core.c > > > new file mode 100644 > > > index 000000000000..0a69ec376c51 > > > --- /dev/null > > > +++ b/drivers/pwm/pwm-microchip-core.c > > > @@ -0,0 +1,509 @@ > > > +// SPDX-License-Identifier: GPL-2.0 > > > +/* > > > + * corePWM driver for Microchip "soft" FPGA IP cores. > > > + * > > > + * Copyright (c) 2021-2023 Microchip Corporation. All rights reserve= d. > > > + * Author: Conor Dooley > > > + * Documentation: > > > + * https://www.microsemi.com/document-portal/doc_download/1245275-co= repwm-hb > > > + * > > > + * Limitations: > > > + * - If the IP block is configured without "shadow registers", all r= egister > > > + * writes will take effect immediately, causing glitches on the ou= tput. > > > + * If shadow registers *are* enabled, a write to the "SYNC_UPDATE"= register > > > + * notifies the core that it needs to update the registers definin= g the > > > + * waveform from the contents of the "shadow registers". > >=20 > > You only write once to the sync update register (i.e. during probe). So > > that register specifies that a period should be completed before a new > > setting becomes active, right? >=20 > Correct. >=20 > > Even with sync update this is still racy, right? >=20 > I assume the period ticking over as we are updating the values is your > concern here. I'm not sure that there's all that much we can do about > that, so I guess I shall update the comment. Yes, that's what I had in mind. > Perhaps writing out period_steps and prescale should be done after > computing the new duty cycle to reduce the possible window since that > may require an expensive division on a 32-bit arch? *nod* > > > + * - The IP block has no concept of a duty cycle, only rising/fallin= g edges of > > > + * the waveform. Unfortunately, if the rising & falling edges regi= sters have > > > + * the same value written to them the IP block will do whichever o= f a rising > > > + * or a falling edge is possible. I.E. a 50% waveform at twice the= requested > > > + * period. Therefore to get a 0% waveform, the output is set the m= ax high/low > > > + * time depending on polarity. > > > + * If the duty cycle is 0%, and the requested period is less than = the > > > + * available period resolution, this will manifest as a ~100% wave= form (with > > > + * some output glitches) rather than 50%. > >=20 > > The last paragraph refers to negedge =3D 0, posedge =3D 0 and period_st= eps =3D > > 0? >=20 > Yes. Although, I did some poking around with it just now & that actually > only happens if prescale is also 0. > If it is non-zero, get to see some other "interesting behaviour" where > the period becomes gigantic - for example @ prescale =3D 0x3, the period > becomes about a quarter of a second w/ a 50% duty cycle. clk_rate is > 62.5 MHz. I'd need to dig out the RTL to justify that one! >=20 > I've just gone and made apply() return -EINVAL for this, which the > subsystem does for requests of zero periods. Sounds right. > > > + * - The PWM period is set for the whole IP block not per channel. T= he driver > > > + * will only change the period if no other PWM output is enabled. > > > + */ > >=20 > > > +static void mchp_core_pwm_enable(struct pwm_chip *chip, struct pwm_d= evice *pwm, > > > + bool enable, u64 period) > > > +{ > > > + struct mchp_core_pwm_chip *mchp_core_pwm =3D to_mchp_core_pwm(chip); > > > + u8 channel_enable, reg_offset, shift; > > > + > > > + /* > > > + * There are two adjacent 8 bit control regs, the lower reg controls > > > + * 0-7 and the upper reg 8-15. Check if the pwm is in the upper reg > > > + * and if so, offset by the bus width. > > > + */ > > > + reg_offset =3D MCHPCOREPWM_EN(pwm->hwpwm >> 3); > > > + shift =3D pwm->hwpwm & 7; > > > + > > > + channel_enable =3D readb_relaxed(mchp_core_pwm->base + reg_offset); > > > + channel_enable &=3D ~(1 << shift); > > > + channel_enable |=3D (enable << shift); > > > + > > > + writel_relaxed(channel_enable, mchp_core_pwm->base + reg_offset); > > > + mchp_core_pwm->channel_enabled &=3D ~BIT(pwm->hwpwm); > > > + mchp_core_pwm->channel_enabled |=3D enable << pwm->hwpwm; > > > + > > > + /* > > > + * Notify the block to update the waveform from the shadow register= s. > > > + * The updated values will not appear on the bus until they have be= en > > > + * applied to the waveform at the beginning of the next period. > > > + * This is a NO-OP if the channel does not have shadow registers. > > > + */ > >=20 > > The code doesn't match the comment. I think that is a relict from the > > times when we thought that a trigger was necessary to update the > > operating settings from the shadow registers? >=20 > Yeah, I read this back to myself before sending v15 & thought that it > didn't need to be changed. I think removing the first line should go. good. > > > +static u64 mchp_core_pwm_calc_duty(const struct pwm_state *state, u6= 4 clk_rate, > > > + u8 prescale, u8 period_steps) > > > +{ > > > + u64 duty_steps, tmp; > > > + > > > + /* > > > + * Calculate the duty cycle in multiples of the prescaled period: > > > + * duty_steps =3D duty_in_ns / step_in_ns > > > + * step_in_ns =3D (prescale * NSEC_PER_SEC) / clk_rate > > > + * The code below is rearranged slightly to only divide once. > > > + */ > > > + tmp =3D (prescale + 1) * NSEC_PER_SEC; > > > + duty_steps =3D mul_u64_u64_div_u64(state->duty_cycle, clk_rate, tmp= ); > > > + > > > + return duty_steps; > > > +} > > > + > > > +static void mchp_core_pwm_apply_duty(struct pwm_chip *chip, struct p= wm_device *pwm, > > > + const struct pwm_state *state, u64 duty_steps, > > > + u16 period_steps) > > > +{ > > > + struct mchp_core_pwm_chip *mchp_core_pwm =3D to_mchp_core_pwm(chip); > > > + u8 posedge, negedge; > > > + u8 first_edge =3D 0, second_edge =3D duty_steps; > > > + > > > + /* > > > + * Setting posedge =3D=3D negedge doesn't yield a constant output, > > > + * so that's an unsuitable setting to model duty_steps =3D 0. > > > + * In that case set the unwanted edge to a value that never > > > + * triggers. > > > + */ > > > + if (duty_steps =3D=3D 0) > > > + first_edge =3D period_steps + 1; > > > + > > > + if (state->polarity =3D=3D PWM_POLARITY_INVERSED) { > > > + negedge =3D first_edge; > > > + posedge =3D second_edge; > > > + } else { > > > + posedge =3D first_edge; > > > + negedge =3D second_edge; > > > + } > > > + > > > + writel_relaxed(posedge, mchp_core_pwm->base + MCHPCOREPWM_POSEDGE(p= wm->hwpwm)); > > > + writel_relaxed(negedge, mchp_core_pwm->base + MCHPCOREPWM_NEGEDGE(p= wm->hwpwm)); > >=20 > > Is this racy with sync update implemented in the firmware? A comment > > about how the sync update is implemented would be good. >=20 > Unless this is a different fear of racing, see above. OK, so a suitable comment would be along the lines of: /* * Set the sync bit which ensures that periods that already * started are completed unaltered. At each counter reset event * the values are updated from the shadow registers. */ =20 > > > +} > > > + > > > +static int mchp_core_pwm_calc_period(const struct pwm_state *state, = unsigned long clk_rate, > > > + u16 *prescale, u16 *period_steps) > > > +{ > > > + u64 tmp; > > > + u32 remainder; > > > + > > > + /* > > > + * Calculate the period cycles and prescale values. > > > + * The registers are each 8 bits wide & multiplied to compute the p= eriod > > > + * using the formula: > > > + * (prescale + 1) * (period_steps + 1) > > > + * period =3D ------------------------------------- > > > + * clk_rate > > > + * so the maximum period that can be generated is 0x10000 times the > > > + * period of the input clock. > > > + * However, due to the design of the "hardware", it is not possible= to > > > + * attain a 100% duty cycle if the full range of period_steps is us= ed. > > > + * Therefore period_steps is restricted to 0xfe and the maximum mul= tiple > > > + * of the clock period attainable is (0xff + 1) * (0xfe + 1) =3D 0x= ff00 > > > + * > > > + * The prescale and period_steps registers operate similarly to > > > + * CLK_DIVIDER_ONE_BASED, where the value used by the hardware is t= hat > > > + * in the register plus one. > > > + * It's therefore not possible to set a period lower than 1/clk_rat= e, so > > > + * if tmp is 0, abort. Without aborting, we will set a period that = is > > > + * greater than that requested and, more importantly, will trigger = the > > > + * neg-/pos-edge issue described in the limitations. > > > + */ > > > + tmp =3D mul_u64_u64_div_u64(state->period, clk_rate, NSEC_PER_SEC); > > > + if (!tmp) > > > + return -EINVAL; > > > + > > > + if (tmp >=3D MCHPCOREPWM_PERIOD_MAX) { > > > + *prescale =3D MCHPCOREPWM_PRESCALE_MAX; > > > + *period_steps =3D MCHPCOREPWM_PERIOD_STEPS_MAX; > > > + > > > + return 0; > > > + } > > > + > > > + /* > > > + * There are multiple strategies that could be used to choose the > > > + * prescale & period_steps values. > > > + * Here the idea is to pick values so that the selection of duty cy= cles > > > + * is as finegrain as possible. > > > + * This "optimal" value for prescale can be calculated using the ma= ximum > > > + * permitted value of period_steps, 0xfe. > > > + * > > > + * period * clk_rate > > > + * prescale =3D ------------------------- - 1 > > > + * NSEC_PER_SEC * (0xfe + 1) > > > + * > > > + * However, we are purely interested in the integer upper bound of = this > > > + * calculation, so this division should be rounded up before subtra= cting > > > + * 1 > > > + * > > > + * period * clk_rate > > > + * ------------------- was precomputed as `tmp` > > > + * NSEC_PER_SEC > > > + */ > > > + *prescale =3D DIV64_U64_ROUND_UP(tmp, MCHPCOREPWM_PERIOD_STEPS_MAX = + 1) - 1; > >=20 > > If state->period * clk_rate is 765000000001 you get tmp =3D 765 and then > > *prescale =3D 2. However roundup(765000000001 / (1000000000 * 255)) - 1= is > > 3. The problem here is that you're rounding down in the calculation of > > tmp. Of course this is constructed because 765000000001 is prime, but > > I'm sure you get the point :-) >=20 > Hold that thought for a moment.. OK, so the correction below is to make up for the wrong rounding here. I'd like to have that in a comment. Otherwise it wasn't clear to me. =20 > > Also we know that tmp is < 0xff00, so we don't need a 64 bit division > > here. >=20 > Neither here nor below, true. >=20 > > > + /* > > > + * Because 0xff is not a permitted value some error will seep into = the > > > + * calculation of prescale as prescale grows. Specifically, this er= ror > > > + * occurs where the remainder of the prescale calculation is less t= han > > > + * prescale. > > > + * For small values of prescale, only a handful of values will need > > > + * correction, but overall this applies to almost half of the valid > > > + * values for tmp. > > > + * > > > + * To keep the algorithm's decision making consistent, this case is > > > + * checked for and the simple solution is to, in these cases, > > > + * decrement prescale and check that the resulting value of period_= steps > > > + * is valid. > > > + * > > > + * period_steps can be computed from prescale: > > > + * period * clk_rate > > > + * period_steps =3D ----------------------------- - 1 > > > + * NSEC_PER_SEC * (prescale + 1) > > > + * > > > + */ > > > + div_u64_rem(tmp, (MCHPCOREPWM_PERIOD_STEPS_MAX + 1), &remainder); > > > + if (remainder < *prescale) { > > > + u16 smaller_prescale =3D *prescale - 1; > > > + > > > + *period_steps =3D div_u64(tmp, smaller_prescale + 1) - 1; > > > + if (*period_steps < 255) { > > > + *prescale =3D smaller_prescale; > > > + > > > + return 0; > > > + } > > > + } >=20 > ...so in your prime case above, we would initially compute a prescale > value that is too large, and then wind up hitting the test of the > remainder here, thereby realising that the smaller prescale value is a > better fit? > Perhaps that's not an acceptable way to handle the issue though. IMHO it is, but the comment explaining needs some improvement. It should also make clear why rounding cannot lead to prescale - 2 being a good/better candidate. (I haven't thought it through, maybe needs some improvement?) I wonder if the computation can find all improvements given that it only used tmp, but not ->period and clk_rate?! =20 > > I don't understand that part. It triggers for tmp =3D 511. So you prefer > >=20 > > prescale =3D 1 > > period_steps =3D 254 > >=20 > > yielding period =3D 510 / clkrate over > >=20 > > prescale =3D 2 > > period_steps =3D 170 > >=20 > > yielding 513 / clkrate. I wonder why. I missed the -=3D 1 in my example. So it's: It triggers for tmp =3D 511. So you prefer prescale =3D 1 period_steps =3D 254 yielding period =3D 510 / clkrate over prescale =3D 2 period_steps =3D 169 yielding 510 / clkrate. Here it's obvious that the former is the better one. But I wonder why the former isn't found instantly. Wouldn't *prescale =3D tmp / (MCHPCOREPWM_PERIOD_STEPS_MAX + 1) - 1 give a better approximation in general? (Of course with an additional check that *prescale >=3D 0 then.)=20 =2E.. thinking a bit ... yes, I think that's true: If you pick *prescale =3D tmp / (MCHPCOREPWM_PERIOD_STEPS_MAX + 1) - 1, then for each period_steps value =E2=89=A4 254 we have: (*prescale + 1) * (period_steps + 1) =E2=89=A4 (*prescale + 1) * 255 =E2=89=A4 (tmp // (MCHPCOREPWM_PERIOD_STEPS_MAX + 1)) * 255 =E2=89=A4 (tmp // 255) * 255 =E2=89=A4 (tmp / 255) * 255 =3D tmp and you can use the maximal period_steps =3D 0xfe. (Only for tmp < 255 that isn't possible, up to you if you refuse these or pick a smaller value for period_steps.) > Because 513 > 511 & 254 > 170! > Is the aim not to produce a period that is less than or equal to that > requested? The aim of this driver is to pick a prescale/period_steps > combo that satisfies that constraint, while also trying to maximise the > "finegrainness" of the duty cycle. > The latter should be stated in a comment above. ack. I also wonder if such a change breaks the other assumptions a consumer might have. I'll spend some more cycles about that in the next round :-) > > Also tmp =3D 511 is the only value > > where this triggers. There is a mistake somewhere (maybe on my side). >=20 > It should trigger for any value 255 * n < x < 256 * n, no? > Say for tmp of 767: > *prescale =3D DIV64_U64_ROUND_UP(767, 254 + 1) - 1 =3D DIV64_U64_ROUND_UP= (3.00784...) - 1 =3D 3 > remainder =3D 0.00784.. * (254 + 1) =3D 2 >=20 > Am I going nuts? Wouldn't be the first time that I've made a hames of > things here, there are 16 versions for a reason after all. Ah, I had a bogus break statement in my python test code. So it's me who got it wrong. Best regards Uwe --=20 Pengutronix e.K. | Uwe Kleine-K=C3=B6nig = | Industrial Linux Solutions | https://www.pengutronix.de/ | --yrwz5jovccfyp3yp Content-Type: application/pgp-signature; name="signature.asc" -----BEGIN PGP SIGNATURE----- iQEzBAABCgAdFiEEP4GsaTp6HlmJrf7Tj4D7WH0S/k4FAmQ1igQACgkQj4D7WH0S /k47sQf/ek3qewsUMx1gwJ/Ia8XK3nd0oLwSLQcx2D80kMF4llRt3RuEyy9K3v9X HckBI+enRqKqGh1kGWP079RMoUM6azlV0Njb/Plp49Yl42A6YIaSRXbWIgkQwQIh jfSc4r/MiCmhrJ1YBWsl07pjhzXrgWaDesOZatoSow2tLH/CtxDBQJZJ/PHrFHcd Dis82tdNPPWiHhq1rDMymh8Jg5ZwhxVwWDTBd0BX9uMkRRsYEwqtcg2pk4F08tiR 9pr6p5Vgvu42WPeqtPw1N18l5qPeN1B1XOlw3XIIMdM8UaaBKU/Xh9VRpL/NzZ/a 8XJ4RU9yEsbLAO6GdNydd3JHTHlK9w== =F8He -----END PGP SIGNATURE----- --yrwz5jovccfyp3yp--