Hi, On Tue, Jun 14, 2022 at 7:28 PM Julius Werner <jwerner@xxxxxxxxxxxx> wrote: > > Sorry, wrong email for Krzysztof. > > On Tue, Jun 14, 2022 at 7:25 PM Julius Werner <jwerner@xxxxxxxxxxxx> wrote: > > > > We (Chromium OS) have been trying to find a way to pass LPDDR memory > > chip information that is available to the firmware through the FDT > > (mostly for userspace informational purposes, for now). We have been > > using and expanding the existing "jedec,lpddr2" and "jedec,lpddr3" > > bindings for this (e.g. [1]). The goal is to be able to identify the > > memory layout of the system (how the parts look like, how they're tied > > together, how much capacity there is in total) as accurately as > > possible from software-probed values. > > > > The existing bindings contain the fields "io-width" and "density", > > which is terminology directly matching what an LPDDR chip can report > > to firmware through the "Mode Register" interface, specifically MR8. > > (The LPDDR specs describing this are not public, but you can see the > > register definitions in most LPDDR chip datasheets that can be > > randomly found online, e.g. [2] page 37.) The code in > > drivers/memory/of_memory.c also suggests that these are supposed to > > directly correspond to the MR8 values read from the chip, since when > > of_lpddr2_get_info() copies the device tree values into struct > > lpddr2_info, it encodes them in a format that directly matches the > > mode register bit field patterns. > > > > The problem with this is that each individual LPDDR chip has its own > > set of mode registers (per rank) that only describe the density of > > that particular chip (rank). The host memory controller may have > > multiple channels (each of which is basically an entirely separate set > > of physical LPDDR pins on the board), a single channel may be > > connected to multiple LPDDR chips (e.g. if the memory controller has > > an outgoing 32-bit channel, that channel could be tied to two 16-bit > > LPDDR chips by tying the low 16 bits to one and the high 16 bits to > > the other), and then each of those chips may offer multiple > > independent ranks (which rank is being accessed at a given time is > > controlled by a separate chip select pin). > > > > So if we just have one "io-width" and one "density" field in the FDT, > > there's no way to figure out how much memory there's actually > > connected in total, because that only describes a single LPDDR chip. > > Worse, there may be chips where different ranks have different > > densities (e.g. a 6GB dual-rank chip with one 4GB and one 2GB rank), > > and different channels could theoretically be connected to chips of > > completely different manufacturers. > > > > We need to be able to report the information that's currently encoded > > in the "jedec,lpddr2" binding separately for each channel+rank > > combination, and we need to be able to tell how many LPDDR chips are > > combined under a single memory channel. For the former, I'd suggest > > creating a separate FDT node for each channel, and then creating > > subnodes under those for each rank that implement the binding. For the > > latter, I would suggest adding a new property "channel-io-width" which > > describes the width of the channel from the host memory controller's > > point of view, so that you can divide that property by the already > > existing "io-width" property to figure out how many parts are tied > > together in series in a single channel. The final layout, then, would > > look something like this: > > > > lpddr2-channel0 { > > rank0 { > > compatible = "jedec,lpddr2"; > > density = <2048>; > > channel-io-width = <32>; > > io-width = <16>; > > }; > > rank1 { > > compatible = "jedec,lpddr2"; > > density = <1024>; > > channel-io-width = <32>; > > io-width = <16>; > > }; > > }; Two comments about the above: 1. It seems like the top-level node should have a compatible of some type. Without that I guess you're just relying on people to find it based on the name of the node? 2. Why not put the `channel-io-width` property in the channel? Then you don't need to repeat it for each rank that's under the channel? If you do that then I think you're actually not making _any_ modifications to the existing lpddr2 bindings, right? You're just adding a new top-level node that would include the memory nodes under it. Although maybe some extra things you might want to add that would affect the nodes in it: 1. In the above the two ranks are in series, right? ...with a chip select to select rank0 vs rank1? From how SPI works I'd expect that to be represented using "reg", AKA: lpddr2-channel0 { compatible = "jdec,lpddr2-channel"; #address-cells = <1>; #size-cells = <0>; channel-io-width = <32>; rank@0 { reg = <0>; compatible = "jedec,lpddr2"; density = <2048>; io-width = <16>; }; rank@1 { reg = <1>; compatible = "jedec,lpddr2"; density = <1024>; io-width = <16>; }; }; 2. I guess if you had two things in parallel you'd want to know how? Maybe if you had 4 8-bit chips connected to a 32-bit channel maybe it'd look like this: lpddr2-channel0 { compatible = "jdec,lpddr2-channel"; #address-cells = <2>; #size-cells = <0>; channel-io-width = <32>; rank@0,0 { reg = <0 0>; compatible = "jedec,lpddr2"; density = <2048>; io-width = <8>; }; rank@0,7 { reg = <0 7>; compatible = "jedec,lpddr2"; density = <2048>; io-width = <8>; }; rank@0,15 { reg = <0 15>; compatible = "jedec,lpddr2"; density = <2048>; io-width = <8>; }; rank@0,23 { reg = <0 23>; compatible = "jedec,lpddr2"; density = <2048>; io-width = <8>; }; }; ...and I guess you could have things that include serial and parallel hookups... > > lpddr2-channel0 { nit that you define "channel0" twice in your example. > > rank0 { > > compatible = "jedec,lpddr2"; > > density = <2048>; > > channel-io-width = <32>; > > io-width = <16>; > > }; > > rank1 { > > compatible = "jedec,lpddr2"; > > density = <1024>; > > channel-io-width = <32>; > > io-width = <16>; > > }; > > }; > > > > This would be describing a dual-channel, dual-rank layout where each > > 32-bit channel is connected to two 16-bit LPDDR chips in series. The > > total capacity would be (2048 Mbits * (32/16) chips + 1024 Mbits * > > (32/16) chips) * 2 channels = 12Gbits. Just to make sure I'm understanding things: in your hypothetical example we're effectively wasting half of the SDRAM bandwidth of the controller, right? So while what you describe is legal you'd get a much more performant system by hooking the two big chips in parallel on one channel and the two small chips in parallel on the other channel. That would effectively give you a 64-bit wide bus as opposed to the 32-bit wide bus that you describe. > > Does this seem reasonable? If nobody has any objections, I can draft > > up a real patch to change the respective bindings. (The two existing > > uses in platform device trees would stay how they are until the > > respective platform maintainers choose to update them, since only they > > would know the exact configuration. They wouldn't technically violate > > the changed binding since they still contain the same properties > > (other than "channel-io-width" which could be declared optional), but > > they wouldn't represent the total memory layout.) > > > > (Also, btw, would it make sense to use this opportunity to combine the > > "jedec,lpddr2" and "jedec,lpddr3" bindings into a single document? > > They contain all the same properties and I think it makes sense to > > keep them in sync, so duplicating the documentation is just > > unnecessary maintenance overhead. I would also like to add a > > "jedec,lpddr4" binding that has the same properties.) I'm happy to let others chime in, but one way to do this would be to put the super common properties (density, width, etc) in a common file and have it "included" by everyone else. See `bindings/spi/spi-controller.yaml` and then see how all the SPI controllers "reference" that. Once you have that you could create a "jedec,lpddr4" bindings that at least contained the common bits even if it didn't contain all the LPDDR4-specific details and later someone could add in all those extra details. > > [1] https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/Documentation/devicetree/bindings/memory-controllers/ddr/jedec,lpddr2.yaml > > [2] https://media-www.micron.com/-/media/client/global/documents/products/data-sheet/dram/mobile-dram/low-power-dram/lpddr2/2gb_automotive_lpddr2_u89n.pdf