1 /* SPDX-License-Identifier: GPL-2.0-only */
3 #include <bootblock_common.h>
4 #include <cpu/x86/tsc.h>
5 #include <psp_verstage/psp_transfer.h>
11 * Verstage on PSP uses the weak timestamp_tick_freq_mhz implementation returning 1, so the
12 * unit of the timestamps in the transfer buffer is microseconds. The x86 side uses the TSC
13 * rate as reference for the timestamps, so the values from the PSP transfer buffer need to be
14 * multiplied by the TSC frequency in MHz.
16 static uint64_t timestamp_from_usec(uint64_t usec
)
18 return usec
* tsc_freq_mhz();
21 void boot_with_psp_timestamp(uint64_t base_timestamp
)
23 const struct transfer_info_struct
*info
= (const struct transfer_info_struct
*)
24 (void *)(uintptr_t)_transfer_buffer
;
26 if (!transfer_buffer_valid(info
) || info
->timestamp
== 0)
29 replay_transfer_buffer_cbmemc();
32 * info->timestamp is PSP's timestamp (in microseconds)
33 * when x86 processor is released.
35 uint64_t psp_last_ts_usec
= info
->timestamp
;
38 struct timestamp_table
*psp_ts_table
=
39 (struct timestamp_table
*)(void *)
40 ((uintptr_t)_transfer_buffer
+ info
->timestamp_offset
);
41 /* new base_timestamp will be offset for all PSP timestamps. */
42 base_timestamp
-= timestamp_from_usec(psp_last_ts_usec
);
44 for (i
= 0; i
< psp_ts_table
->num_entries
; i
++) {
45 struct timestamp_entry
*tse
= &psp_ts_table
->entries
[i
];
47 * We ignore the time between x86 processor release and bootblock.
48 * Since timestamp_add subtracts base_time, we first add old base_time
49 * to make it absolute then add base_timestamp again since
50 * it'll be a new base_time.
52 * Verstage on PSP uses a 1 microsecond timestamp granularity while the x86
53 * part of coreboot uses the TSC tick time as granularity, so this needs to be
56 tse
->entry_stamp
= timestamp_from_usec(psp_ts_table
->base_time
+
57 tse
->entry_stamp
) + base_timestamp
;
60 bootblock_main_with_timestamp(base_timestamp
, psp_ts_table
->entries
,
61 psp_ts_table
->num_entries
);