Memory, SPD and Coreboot on Intel Macbooks

January 14, 2025

Updated: January 29, 2025

Introduction

Apple was one of the first manufacturers that started soldering its memory on their laptops. This has an effect on how the boot firmware handles its SPD configurations. With non-soldered ram the SPD configuration is stored on the memory modules themselves, with ram soldered the needed information is in the boot firmware. This brings a challenge when trying to install and boot with alternate firmware.

In this article we are going to look at how to extract to right information for memory management from a Macbook Pro 10,1 (A1398) and implement it in Coreboot, but the information should apply to similar macbooks from that era. Apple uses many different kinds of RAM chips in this model, as well as other macbooks. This makes it difficult to easily support a Macbook in Coreboot, since all of these different memory configurations have to be supported.

The Data

To be able to support the memory configuration, we would have to know what kind of memory is in this model. We will use a tool called “Inteltool” (part of the coreboot utils) to extract the information. To use inteltool on these macbooks it's easiest to boot from a Debian GNU/Linux Live USB and install the coreboot-utils though the apt package manager, copy the inteltool program from a different GNU/Linux system that has the coreboot-utils package installed or extract it directly from the coreboot-utils .deb package. When we run inteltool -m we will get a long output, but we will focus on the following part of the output:

/* CH0S0: 4096 MiB  */
/* CH1S0: 4096 MiB  */
/* SPD matching current mode:  */
/* CH0S0  */
00: 92 11 0b 03 03 00 00 09 03 52 01 08 0a 00 80 00 
10: 6e 78 6e 32 6e 11 18 81 00 05 3c 3c 00 f0 00 00 
20: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
30: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 65 00 
40: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
50: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
60: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
70: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 27 06 
80: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
90: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
a0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
b0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
c0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
d0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
e0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
f0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 

/* CH1S0  */
00: 92 11 0b 03 03 00 00 09 03 52 01 08 0a 00 80 00 
10: 6e 78 6e 32 6e 11 18 81 00 05 3c 3c 00 f0 00 00 
20: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
30: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 65 00 
40: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
50: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
60: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
70: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 27 06 
80: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
90: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
a0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
b0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
c0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
d0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
e0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
f0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
Even though this is this is the exact SPD config of the modules, does not mean that it can boot with this configuration. We will get to this at a later part.

The early initialization

Before we go any further we will look at what is already supported. When we go to the repository we can look at the SPD configurations and the early_init.c. Which is currently this: (relevant part of) early_init.c
static unsigned int get_spd_index(void)
{
	const int spd_gpio_vector[] = {71, 70, 69, 68, -1};
	int ramcfg = get_gpios(spd_gpio_vector);
	const int spd_map[] = {-1, 0, -1, -1, -1, -1, 2, 1, -1, 2, -1, -1, -1, -1, -1, -1};
	int spd_index = spd_map[ramcfg];
	/*
	 * GPIO68 GPIO69 GPIO70 GPIO71 Memory           Supported
	 * 0      0      0      0      4G Hynix 1600S   No
	 * 0      0      0      1      1G Samsung 1600  Yes
	 * 0      0      1      0      4G Samsung 1600S No
	 * 0      0      1      1      1G Hynix 1600    No
	 * 0      1      0      0      4G Elpida 1600S  No
	 * 0      1      0      1      2G Samsung 1600  No
	 * 0      1      1      0      2G Samsung 1333  Yes
	 * 0      1      1      1      2G Hynix 1600    Yes
	 * 1      0      0      0      4G Samsung 1600  No
	 * 1      0      0      1      4G Hynix 1600    Yes
	 * 1      0      1      0      2G Elpida 1600S  No
	 * 1      0      1      1      2G Elpida 1600   No
	 * 1      1      0      0      4G Elpida 1600   No
	 * 1      1      0      1      2G Samsung 1600S No
	 * 1      1      1      0      2G Hynix 1600S   No
	 */
	if (spd_index == -1) {
		die("Unsupported memory, RAMCFG=%d. "
			"Please contact the coreboot mailing list\n", ramcfg);
	}
	return spd_index;
}
Originally this information is extracted from the schematics of this board (820-3332-a, page 5). It is possible to support RAM configuration for boards that don't have schematics but it won't be possible to know how many different configurations there are and what brand/speed chips are connected to the GPIO values and SPD mappings. Some memory configurations are already supported and are connnected to an SPD map also in the repository. At this point there are only three spd maps in the repository, even though we have 4 supported memory configurations. How is that possible? Well in this case the model that used 2G Samsung 1333 memory chips uses the same SPD map as the 4G hynix 1600. This seems very odd but is the case sometimes. It is also possible the already existing naming scheme may not be entirely correct. So how does our memory identify itself and how can we add the right SPD config or maybe point to one that already exists? To find this out we go to the next chapter: GPIO Values

GPIO Values

We can get all the SPD mappings there are, but how does the firmware know which one to pick for machine to boot? The Boot firmware will look at certain GPIO values that are present in the hardware to pick to right SPD mapping. As you can already see in the early_init.c, there are values for GPIO68, GPIO69, GPIO70 and GPIO71. A combination of 1s and 0s that will be unique for every memory configuration. Now we need to find out what the GPIO68-71 values are for our machine to look if it’s either already supported or support for what configuration we need to add. For this we need to run inteltool -g. Again we will get a long output but here is a snippet to focuses on the values we need:

gpiobase+0x0030: 0x13ff80fe (GPIO_USE_SEL2)
gpiobase+0x0034: 0x1f04ffe2 (GP_IO_SEL2)
gpiobase+0x0038: 0xfeaf9fc6 (GP_LVL2)
gpiobase+0x003c: 0x00000000 (RESERVED)
gpiobase+0x0040: 0x000006ff (GPIO_USE_SEL3)
gpiobase+0x0044: 0x00000ff0 (GP_IO_SEL3)
gpiobase+0x0048: 0x00000fe0 (GP_LVL3)
gpiobase+0x004c: 0x00000000 (RESERVED)
We need the record that includes the GPIO values for our memory. When we look at Intel Datasheets for chipsets from this era ( 13.10.16 GP_LVL3—GPIO Level for Input or Output 3 Register) . We can see the value GP_LVL3 included the GPIO64-75 values. Even though GPIO_USE_SEL3 and GPIO_IO_SEL3 also are related to these GPIO values, it does not include the right data for our RAM. So now that we know we need some values from GP_LVL3, we will look at how to extract them. GP_LVL3 has a hexidecimal value. To make it readable we will convert it to binary. When we convert “0x00000fe0” to binary we get “111111100000” Now each bit will refer to a GPIO value. Remember that we said this contained GPIO 64 to 75? Well that is exactly how the values correspond. To visualize it. Let’s put each bit to it’s respective GPIO value
GPIO75 = 1
GPIO74 = 1
GPIO73 = 1
GPIO72 = 1
GPIO71 = 1
GPIO70 = 1
GPIO69 = 1
GPIO68 = 0
GPIO67 = 0
GPIO66 = 0
GPIO65 = 0
GPIO64 = 0
Now when we focus on GPIO68 to 71 have the following values: 0 1 1 1. At this point we can compare that to the the values present in early_init.c
 * GPIO68 GPIO69 GPIO70 GPIO71 Memory           Supported
	 * 0      0      0      0      4G Hynix 1600S   No
	 * 0      0      0      1      1G Samsung 1600  Yes
	 * 0      0      1      0      4G Samsung 1600S No
	 * 0      0      1      1      1G Hynix 1600    No
	 * 0      1      0      0      4G Elpida 1600S  No
	 * 0      1      0      1      2G Samsung 1600  No
	 * 0      1      1      0      2G Samsung 1333  Yes
	 * 0      1      1      1      2G Hynix 1600    Yes
	 * 1      0      0      0      4G Samsung 1600  No
	 * 1      0      0      1      4G Hynix 1600    Yes
	 * 1      0      1      0      2G Elpida 1600S  No
	 * 1      0      1      1      2G Elpida 1600   No
	 * 1      1      0      0      4G Elpida 1600   No
	 * 1      1      0      1      2G Samsung 1600S No
	 * 1      1      1      0      2G Hynix 1600S   No
	 
As you can see the GPIO values 0 1 1 1 corresponds to 2G Hynix 1600 which is already supported. This is a quite common memory configuration and seems to be present in most (if not all) I7 8GB models. But for the sake of this exercise, let’s say it is not supported.

Supporting a new configuration

To support a new configuration you normally have to add the unsupported SPD mapping a new hex file, include it in the SPD makefile.mk and link to that file in the const int spd_map in early_init.c. Normally for us that would mean. We create /spd/2g_hynix_1600.spd.hex file. Copy the SPD information we extracted earlier with inteltool -m. Create a new line in spd/makefile.mk with the right filename and change the 8th value in const int spd_map from -1 (unsupported) to 1 (second SPD mapping). Sounds easy, right? Well unfortunately the SPD that we extracted will not work on this machine. Why?

Rank 1 mirroring

When we compare our extracted SPD map and compare it to the one in 2g_hynix_1600.spd.hex, we will see they do not correspond. This is because the memory uses Rank 1 mirroring. Rank 1 mirroring is a design choice and is not present in all memory configurations. For example the 1G Samsung 1600 memory does not use rank 1 mirroring. It may be possible to see which memory uses Rank mirroring and which doesn’t, either physically or in the software. But I haven’t found a conclusive way to find the right information. So it may be a case of trial and error for now. To see the difference between a rank mirrored configuration and without, look at the SPD mappings below:

2g hynix 1600 (rank 1 mirroring enabled)
92 11 0b 03 03 00 00 09 03 52 01 08 0a 00 80 00
6e 78 6e 32 6e 11 18 81 00 05 3c 3c 00 f0 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 65 01
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 27 06
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
2g hynix 1600 (rank 1 mirroring disabled)
92 11 0b 03 03 00 00 09 03 52 01 08 0a 00 80 00 
6e 78 6e 32 6e 11 18 81 00 05 3c 3c 00 f0 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 65 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 27 06 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 
The only difference is that byte 63 is flipped. Without mirroring it has a value of 00 and with mirroring it has a value of 01. With DDR3 modules rank mirroring is always done with the 63rd byte. the following publication from simmtester describes it best:
Byte 63
Address Mapping from Edge Connector to DRAM

For ease of module PCB layout, sometimes “mirror” address mapping is used. “Mirror” address is to flip the address line sequence on the 2nd rank of the module. This byte describes the connection of edge connector pins for address bits to the corresponding input pins of the DDR3 SDRAMs.

Rank 1 Mapping   

Standard            00h
Mirrored            01h 

Last Words

RAM initialization has always been hard in Coreboot, but with soldered ram that doesn't provide the right information to the boot firmware it gets even more complicated. Hopefully this article will provide you with some insight on how this works. When you manage to support a new memory configuration yourself, you can follow ch1p's instructions in his blogpost "On the current state of coreboot on MacBooks" to compile and flash coreboot. Do not hesitate to contact me with any questions.

Updates

January 29, 2025:

Fixed some source and formatting errors and cleared up text about inteltool and and rank mirroring.

Sources

Relevant sources for this article:

https://ch1p.io/coreboot-macbook-support/
https://review.coreboot.org/c/coreboot/+/32673
https://review.coreboot.org/c/coreboot/+/32673/59/src/mainboard/apple/macbookpro10_1/early_init.c
https://www.intel.com/content/dam/www/public/us/en/documents/datasheets/6-chipset-c200-chipset-datasheet.pdf

Both Ch1p and nic3 from the coreboot/libreboot community were a huge help to understand this material. Couldn't have done it without them.

nlnet

Contact information

You can contact mlp [at] privm.org for any questions regarding this project.

This project is funded by nlnet and NGI0 Entrust. Project page.