xref: /linux/drivers/gpu/drm/v3d/v3d_power.c (revision bba2c3615bd6cfee7456d1130f2e6b01b3f4e9ba)
1 // SPDX-License-Identifier: GPL-2.0+
2 /* Copyright (C) 2026 Raspberry Pi */
3 
4 #include <linux/clk.h>
5 
6 #include <drm/drm_print.h>
7 
8 #include "v3d_drv.h"
9 #include "v3d_regs.h"
10 
11 static int
12 v3d_resume_sms(struct v3d_dev *v3d)
13 {
14 	if (v3d->ver < V3D_GEN_71)
15 		return 0;
16 
17 	V3D_SMS_WRITE(V3D_SMS_TEE_CS, V3D_SMS_CLEAR_POWER_OFF);
18 
19 	if (wait_for((V3D_GET_FIELD(V3D_SMS_READ(V3D_SMS_TEE_CS),
20 				    V3D_SMS_STATE) == V3D_SMS_IDLE), 100)) {
21 		drm_err(&v3d->drm, "Failed to power up SMS\n");
22 		return -ETIMEDOUT;
23 	}
24 
25 	v3d_reset_sms(v3d);
26 
27 	return 0;
28 }
29 
30 static int
31 v3d_suspend_sms(struct v3d_dev *v3d)
32 {
33 	if (v3d->ver < V3D_GEN_71)
34 		return 0;
35 
36 	V3D_SMS_WRITE(V3D_SMS_TEE_CS, V3D_SMS_POWER_OFF);
37 
38 	if (wait_for((V3D_GET_FIELD(V3D_SMS_READ(V3D_SMS_TEE_CS),
39 				    V3D_SMS_STATE) == V3D_SMS_POWER_OFF_STATE), 100)) {
40 		drm_err(&v3d->drm, "Failed to power off SMS\n");
41 		return -ETIMEDOUT;
42 	}
43 
44 	return 0;
45 }
46 
47 int v3d_power_suspend(struct device *dev)
48 {
49 	struct drm_device *drm = dev_get_drvdata(dev);
50 	struct v3d_dev *v3d = to_v3d_dev(drm);
51 	int ret;
52 
53 	v3d_irq_disable(v3d);
54 
55 	v3d_clean_caches(v3d);
56 
57 	ret = v3d_suspend_sms(v3d);
58 	if (ret) {
59 		v3d_irq_enable(v3d);
60 		return ret;
61 	}
62 
63 	clk_disable_unprepare(v3d->clk);
64 
65 	return 0;
66 }
67 
68 int v3d_power_resume(struct device *dev)
69 {
70 	struct drm_device *drm = dev_get_drvdata(dev);
71 	struct v3d_dev *v3d = to_v3d_dev(drm);
72 	int ret;
73 
74 	ret = clk_prepare_enable(v3d->clk);
75 	if (ret)
76 		return ret;
77 
78 	ret = v3d_resume_sms(v3d);
79 	if (ret) {
80 		clk_disable_unprepare(v3d->clk);
81 		return ret;
82 	}
83 
84 	v3d_init_hw_state(v3d);
85 	v3d_mmu_set_page_table(v3d);
86 	v3d_irq_enable(v3d);
87 
88 	return 0;
89 }
90