diff options
author | Dave Airlie <airlied@redhat.com> | 2016-06-23 20:34:35 -0400 |
---|---|---|
committer | Dave Airlie <airlied@redhat.com> | 2016-06-23 20:34:35 -0400 |
commit | f762bfda2b81db4c3397a31e92f819818cf04efb (patch) | |
tree | 57c0300d1e7a6592cbd2083ae30f0900b627ab52 | |
parent | c38e80169befdea4a9438cc6c513794d7c50e88f (diff) | |
parent | 13fef095bde04228316046f997eb963285d8532e (diff) |
Merge tag 'sunxi-drm-fixes-for-4.7' of https://git.kernel.org/pub/scm/linux/kernel/git/mripard/linux into drm-fixes
Allwinner sun4i DRM driver fixes
A bunch of fixes that address:
- Compilation errors in various corner cases
- Move to helpers
- Fix the pixel clock computation
- Fix our panel probe
* tag 'sunxi-drm-fixes-for-4.7' of https://git.kernel.org/pub/scm/linux/kernel/git/mripard/linux:
drm: sun4i: do cleanup if RGB output init fails
drm/sun4i: Convert to connector register helpers
drm/sun4i: remove simplefb at probe
drm/sun4i: rgb: panel is an error pointer
drm/sun4i: defer only if we didn't find our panel
drm/sun4i: rgb: Validate the clock rate
drm/sun4i: request exact rates to our parents
drm: sun4i: fix probe error handling
drm: sun4i: print DMA address correctly
drm/sun4i: add COMMON_CLK dependency
-rw-r--r-- | drivers/gpu/drm/sun4i/Kconfig | 2 | ||||
-rw-r--r-- | drivers/gpu/drm/sun4i/sun4i_backend.c | 4 | ||||
-rw-r--r-- | drivers/gpu/drm/sun4i/sun4i_dotclock.c | 39 | ||||
-rw-r--r-- | drivers/gpu/drm/sun4i/sun4i_drv.c | 50 | ||||
-rw-r--r-- | drivers/gpu/drm/sun4i/sun4i_rgb.c | 16 | ||||
-rw-r--r-- | drivers/gpu/drm/sun4i/sun4i_tcon.c | 23 |
6 files changed, 88 insertions, 46 deletions
diff --git a/drivers/gpu/drm/sun4i/Kconfig b/drivers/gpu/drm/sun4i/Kconfig index 99510e64e91a..a4b357db8856 100644 --- a/drivers/gpu/drm/sun4i/Kconfig +++ b/drivers/gpu/drm/sun4i/Kconfig | |||
@@ -1,6 +1,6 @@ | |||
1 | config DRM_SUN4I | 1 | config DRM_SUN4I |
2 | tristate "DRM Support for Allwinner A10 Display Engine" | 2 | tristate "DRM Support for Allwinner A10 Display Engine" |
3 | depends on DRM && ARM | 3 | depends on DRM && ARM && COMMON_CLK |
4 | depends on ARCH_SUNXI || COMPILE_TEST | 4 | depends on ARCH_SUNXI || COMPILE_TEST |
5 | select DRM_GEM_CMA_HELPER | 5 | select DRM_GEM_CMA_HELPER |
6 | select DRM_KMS_HELPER | 6 | select DRM_KMS_HELPER |
diff --git a/drivers/gpu/drm/sun4i/sun4i_backend.c b/drivers/gpu/drm/sun4i/sun4i_backend.c index f7a15c1a93bf..3ab560450a82 100644 --- a/drivers/gpu/drm/sun4i/sun4i_backend.c +++ b/drivers/gpu/drm/sun4i/sun4i_backend.c | |||
@@ -190,7 +190,7 @@ int sun4i_backend_update_layer_buffer(struct sun4i_backend *backend, | |||
190 | /* Get the physical address of the buffer in memory */ | 190 | /* Get the physical address of the buffer in memory */ |
191 | gem = drm_fb_cma_get_gem_obj(fb, 0); | 191 | gem = drm_fb_cma_get_gem_obj(fb, 0); |
192 | 192 | ||
193 | DRM_DEBUG_DRIVER("Using GEM @ 0x%x\n", gem->paddr); | 193 | DRM_DEBUG_DRIVER("Using GEM @ %pad\n", &gem->paddr); |
194 | 194 | ||
195 | /* Compute the start of the displayed memory */ | 195 | /* Compute the start of the displayed memory */ |
196 | bpp = drm_format_plane_cpp(fb->pixel_format, 0); | 196 | bpp = drm_format_plane_cpp(fb->pixel_format, 0); |
@@ -198,7 +198,7 @@ int sun4i_backend_update_layer_buffer(struct sun4i_backend *backend, | |||
198 | paddr += (state->src_x >> 16) * bpp; | 198 | paddr += (state->src_x >> 16) * bpp; |
199 | paddr += (state->src_y >> 16) * fb->pitches[0]; | 199 | paddr += (state->src_y >> 16) * fb->pitches[0]; |
200 | 200 | ||
201 | DRM_DEBUG_DRIVER("Setting buffer address to 0x%x\n", paddr); | 201 | DRM_DEBUG_DRIVER("Setting buffer address to %pad\n", &paddr); |
202 | 202 | ||
203 | /* Write the 32 lower bits of the address (in bits) */ | 203 | /* Write the 32 lower bits of the address (in bits) */ |
204 | lo_paddr = paddr << 3; | 204 | lo_paddr = paddr << 3; |
diff --git a/drivers/gpu/drm/sun4i/sun4i_dotclock.c b/drivers/gpu/drm/sun4i/sun4i_dotclock.c index 3ff668cb463c..5b3463197c48 100644 --- a/drivers/gpu/drm/sun4i/sun4i_dotclock.c +++ b/drivers/gpu/drm/sun4i/sun4i_dotclock.c | |||
@@ -72,14 +72,40 @@ static unsigned long sun4i_dclk_recalc_rate(struct clk_hw *hw, | |||
72 | static long sun4i_dclk_round_rate(struct clk_hw *hw, unsigned long rate, | 72 | static long sun4i_dclk_round_rate(struct clk_hw *hw, unsigned long rate, |
73 | unsigned long *parent_rate) | 73 | unsigned long *parent_rate) |
74 | { | 74 | { |
75 | return *parent_rate / DIV_ROUND_CLOSEST(*parent_rate, rate); | 75 | unsigned long best_parent = 0; |
76 | u8 best_div = 1; | ||
77 | int i; | ||
78 | |||
79 | for (i = 6; i < 127; i++) { | ||
80 | unsigned long ideal = rate * i; | ||
81 | unsigned long rounded; | ||
82 | |||
83 | rounded = clk_hw_round_rate(clk_hw_get_parent(hw), | ||
84 | ideal); | ||
85 | |||
86 | if (rounded == ideal) { | ||
87 | best_parent = rounded; | ||
88 | best_div = i; | ||
89 | goto out; | ||
90 | } | ||
91 | |||
92 | if ((rounded < ideal) && (rounded > best_parent)) { | ||
93 | best_parent = rounded; | ||
94 | best_div = i; | ||
95 | } | ||
96 | } | ||
97 | |||
98 | out: | ||
99 | *parent_rate = best_parent; | ||
100 | |||
101 | return best_parent / best_div; | ||
76 | } | 102 | } |
77 | 103 | ||
78 | static int sun4i_dclk_set_rate(struct clk_hw *hw, unsigned long rate, | 104 | static int sun4i_dclk_set_rate(struct clk_hw *hw, unsigned long rate, |
79 | unsigned long parent_rate) | 105 | unsigned long parent_rate) |
80 | { | 106 | { |
81 | struct sun4i_dclk *dclk = hw_to_dclk(hw); | 107 | struct sun4i_dclk *dclk = hw_to_dclk(hw); |
82 | int div = DIV_ROUND_CLOSEST(parent_rate, rate); | 108 | u8 div = parent_rate / rate; |
83 | 109 | ||
84 | return regmap_update_bits(dclk->regmap, SUN4I_TCON0_DCLK_REG, | 110 | return regmap_update_bits(dclk->regmap, SUN4I_TCON0_DCLK_REG, |
85 | GENMASK(6, 0), div); | 111 | GENMASK(6, 0), div); |
@@ -127,10 +153,14 @@ int sun4i_dclk_create(struct device *dev, struct sun4i_tcon *tcon) | |||
127 | const char *clk_name, *parent_name; | 153 | const char *clk_name, *parent_name; |
128 | struct clk_init_data init; | 154 | struct clk_init_data init; |
129 | struct sun4i_dclk *dclk; | 155 | struct sun4i_dclk *dclk; |
156 | int ret; | ||
130 | 157 | ||
131 | parent_name = __clk_get_name(tcon->sclk0); | 158 | parent_name = __clk_get_name(tcon->sclk0); |
132 | of_property_read_string_index(dev->of_node, "clock-output-names", 0, | 159 | ret = of_property_read_string_index(dev->of_node, |
133 | &clk_name); | 160 | "clock-output-names", 0, |
161 | &clk_name); | ||
162 | if (ret) | ||
163 | return ret; | ||
134 | 164 | ||
135 | dclk = devm_kzalloc(dev, sizeof(*dclk), GFP_KERNEL); | 165 | dclk = devm_kzalloc(dev, sizeof(*dclk), GFP_KERNEL); |
136 | if (!dclk) | 166 | if (!dclk) |
@@ -140,6 +170,7 @@ int sun4i_dclk_create(struct device *dev, struct sun4i_tcon *tcon) | |||
140 | init.ops = &sun4i_dclk_ops; | 170 | init.ops = &sun4i_dclk_ops; |
141 | init.parent_names = &parent_name; | 171 | init.parent_names = &parent_name; |
142 | init.num_parents = 1; | 172 | init.num_parents = 1; |
173 | init.flags = CLK_SET_RATE_PARENT; | ||
143 | 174 | ||
144 | dclk->regmap = tcon->regs; | 175 | dclk->regmap = tcon->regs; |
145 | dclk->hw.init = &init; | 176 | dclk->hw.init = &init; |
diff --git a/drivers/gpu/drm/sun4i/sun4i_drv.c b/drivers/gpu/drm/sun4i/sun4i_drv.c index 76e922bb60e5..257d2b4f3645 100644 --- a/drivers/gpu/drm/sun4i/sun4i_drv.c +++ b/drivers/gpu/drm/sun4i/sun4i_drv.c | |||
@@ -24,34 +24,6 @@ | |||
24 | #include "sun4i_layer.h" | 24 | #include "sun4i_layer.h" |
25 | #include "sun4i_tcon.h" | 25 | #include "sun4i_tcon.h" |
26 | 26 | ||
27 | static int sun4i_drv_connector_plug_all(struct drm_device *drm) | ||
28 | { | ||
29 | struct drm_connector *connector, *failed; | ||
30 | int ret; | ||
31 | |||
32 | mutex_lock(&drm->mode_config.mutex); | ||
33 | list_for_each_entry(connector, &drm->mode_config.connector_list, head) { | ||
34 | ret = drm_connector_register(connector); | ||
35 | if (ret) { | ||
36 | failed = connector; | ||
37 | goto err; | ||
38 | } | ||
39 | } | ||
40 | mutex_unlock(&drm->mode_config.mutex); | ||
41 | return 0; | ||
42 | |||
43 | err: | ||
44 | list_for_each_entry(connector, &drm->mode_config.connector_list, head) { | ||
45 | if (failed == connector) | ||
46 | break; | ||
47 | |||
48 | drm_connector_unregister(connector); | ||
49 | } | ||
50 | mutex_unlock(&drm->mode_config.mutex); | ||
51 | |||
52 | return ret; | ||
53 | } | ||
54 | |||
55 | static int sun4i_drv_enable_vblank(struct drm_device *drm, unsigned int pipe) | 27 | static int sun4i_drv_enable_vblank(struct drm_device *drm, unsigned int pipe) |
56 | { | 28 | { |
57 | struct sun4i_drv *drv = drm->dev_private; | 29 | struct sun4i_drv *drv = drm->dev_private; |
@@ -125,6 +97,22 @@ static struct drm_driver sun4i_drv_driver = { | |||
125 | .disable_vblank = sun4i_drv_disable_vblank, | 97 | .disable_vblank = sun4i_drv_disable_vblank, |
126 | }; | 98 | }; |
127 | 99 | ||
100 | static void sun4i_remove_framebuffers(void) | ||
101 | { | ||
102 | struct apertures_struct *ap; | ||
103 | |||
104 | ap = alloc_apertures(1); | ||
105 | if (!ap) | ||
106 | return; | ||
107 | |||
108 | /* The framebuffer can be located anywhere in RAM */ | ||
109 | ap->ranges[0].base = 0; | ||
110 | ap->ranges[0].size = ~0; | ||
111 | |||
112 | remove_conflicting_framebuffers(ap, "sun4i-drm-fb", false); | ||
113 | kfree(ap); | ||
114 | } | ||
115 | |||
128 | static int sun4i_drv_bind(struct device *dev) | 116 | static int sun4i_drv_bind(struct device *dev) |
129 | { | 117 | { |
130 | struct drm_device *drm; | 118 | struct drm_device *drm; |
@@ -172,6 +160,9 @@ static int sun4i_drv_bind(struct device *dev) | |||
172 | } | 160 | } |
173 | drm->irq_enabled = true; | 161 | drm->irq_enabled = true; |
174 | 162 | ||
163 | /* Remove early framebuffers (ie. simplefb) */ | ||
164 | sun4i_remove_framebuffers(); | ||
165 | |||
175 | /* Create our framebuffer */ | 166 | /* Create our framebuffer */ |
176 | drv->fbdev = sun4i_framebuffer_init(drm); | 167 | drv->fbdev = sun4i_framebuffer_init(drm); |
177 | if (IS_ERR(drv->fbdev)) { | 168 | if (IS_ERR(drv->fbdev)) { |
@@ -187,7 +178,7 @@ static int sun4i_drv_bind(struct device *dev) | |||
187 | if (ret) | 178 | if (ret) |
188 | goto free_drm; | 179 | goto free_drm; |
189 | 180 | ||
190 | ret = sun4i_drv_connector_plug_all(drm); | 181 | ret = drm_connector_register_all(drm); |
191 | if (ret) | 182 | if (ret) |
192 | goto unregister_drm; | 183 | goto unregister_drm; |
193 | 184 | ||
@@ -204,6 +195,7 @@ static void sun4i_drv_unbind(struct device *dev) | |||
204 | { | 195 | { |
205 | struct drm_device *drm = dev_get_drvdata(dev); | 196 | struct drm_device *drm = dev_get_drvdata(dev); |
206 | 197 | ||
198 | drm_connector_unregister_all(drm); | ||
207 | drm_dev_unregister(drm); | 199 | drm_dev_unregister(drm); |
208 | drm_kms_helper_poll_fini(drm); | 200 | drm_kms_helper_poll_fini(drm); |
209 | sun4i_framebuffer_free(drm); | 201 | sun4i_framebuffer_free(drm); |
diff --git a/drivers/gpu/drm/sun4i/sun4i_rgb.c b/drivers/gpu/drm/sun4i/sun4i_rgb.c index ab6494818050..aaffe9e64ffb 100644 --- a/drivers/gpu/drm/sun4i/sun4i_rgb.c +++ b/drivers/gpu/drm/sun4i/sun4i_rgb.c | |||
@@ -54,8 +54,13 @@ static int sun4i_rgb_get_modes(struct drm_connector *connector) | |||
54 | static int sun4i_rgb_mode_valid(struct drm_connector *connector, | 54 | static int sun4i_rgb_mode_valid(struct drm_connector *connector, |
55 | struct drm_display_mode *mode) | 55 | struct drm_display_mode *mode) |
56 | { | 56 | { |
57 | struct sun4i_rgb *rgb = drm_connector_to_sun4i_rgb(connector); | ||
58 | struct sun4i_drv *drv = rgb->drv; | ||
59 | struct sun4i_tcon *tcon = drv->tcon; | ||
57 | u32 hsync = mode->hsync_end - mode->hsync_start; | 60 | u32 hsync = mode->hsync_end - mode->hsync_start; |
58 | u32 vsync = mode->vsync_end - mode->vsync_start; | 61 | u32 vsync = mode->vsync_end - mode->vsync_start; |
62 | unsigned long rate = mode->clock * 1000; | ||
63 | long rounded_rate; | ||
59 | 64 | ||
60 | DRM_DEBUG_DRIVER("Validating modes...\n"); | 65 | DRM_DEBUG_DRIVER("Validating modes...\n"); |
61 | 66 | ||
@@ -87,6 +92,15 @@ static int sun4i_rgb_mode_valid(struct drm_connector *connector, | |||
87 | 92 | ||
88 | DRM_DEBUG_DRIVER("Vertical parameters OK\n"); | 93 | DRM_DEBUG_DRIVER("Vertical parameters OK\n"); |
89 | 94 | ||
95 | rounded_rate = clk_round_rate(tcon->dclk, rate); | ||
96 | if (rounded_rate < rate) | ||
97 | return MODE_CLOCK_LOW; | ||
98 | |||
99 | if (rounded_rate > rate) | ||
100 | return MODE_CLOCK_HIGH; | ||
101 | |||
102 | DRM_DEBUG_DRIVER("Clock rate OK\n"); | ||
103 | |||
90 | return MODE_OK; | 104 | return MODE_OK; |
91 | } | 105 | } |
92 | 106 | ||
@@ -203,7 +217,7 @@ int sun4i_rgb_init(struct drm_device *drm) | |||
203 | int ret; | 217 | int ret; |
204 | 218 | ||
205 | /* If we don't have a panel, there's no point in going on */ | 219 | /* If we don't have a panel, there's no point in going on */ |
206 | if (!tcon->panel) | 220 | if (IS_ERR(tcon->panel)) |
207 | return -ENODEV; | 221 | return -ENODEV; |
208 | 222 | ||
209 | rgb = devm_kzalloc(drm->dev, sizeof(*rgb), GFP_KERNEL); | 223 | rgb = devm_kzalloc(drm->dev, sizeof(*rgb), GFP_KERNEL); |
diff --git a/drivers/gpu/drm/sun4i/sun4i_tcon.c b/drivers/gpu/drm/sun4i/sun4i_tcon.c index 9f19b0e08560..652385f09735 100644 --- a/drivers/gpu/drm/sun4i/sun4i_tcon.c +++ b/drivers/gpu/drm/sun4i/sun4i_tcon.c | |||
@@ -425,11 +425,11 @@ static struct drm_panel *sun4i_tcon_find_panel(struct device_node *node) | |||
425 | 425 | ||
426 | remote = of_graph_get_remote_port_parent(end_node); | 426 | remote = of_graph_get_remote_port_parent(end_node); |
427 | if (!remote) { | 427 | if (!remote) { |
428 | DRM_DEBUG_DRIVER("Enable to parse remote node\n"); | 428 | DRM_DEBUG_DRIVER("Unable to parse remote node\n"); |
429 | return ERR_PTR(-EINVAL); | 429 | return ERR_PTR(-EINVAL); |
430 | } | 430 | } |
431 | 431 | ||
432 | return of_drm_find_panel(remote); | 432 | return of_drm_find_panel(remote) ?: ERR_PTR(-EPROBE_DEFER); |
433 | } | 433 | } |
434 | 434 | ||
435 | static int sun4i_tcon_bind(struct device *dev, struct device *master, | 435 | static int sun4i_tcon_bind(struct device *dev, struct device *master, |
@@ -490,7 +490,11 @@ static int sun4i_tcon_bind(struct device *dev, struct device *master, | |||
490 | return 0; | 490 | return 0; |
491 | } | 491 | } |
492 | 492 | ||
493 | return sun4i_rgb_init(drm); | 493 | ret = sun4i_rgb_init(drm); |
494 | if (ret < 0) | ||
495 | goto err_free_clocks; | ||
496 | |||
497 | return 0; | ||
494 | 498 | ||
495 | err_free_clocks: | 499 | err_free_clocks: |
496 | sun4i_tcon_free_clocks(tcon); | 500 | sun4i_tcon_free_clocks(tcon); |
@@ -522,12 +526,13 @@ static int sun4i_tcon_probe(struct platform_device *pdev) | |||
522 | * Defer the probe. | 526 | * Defer the probe. |
523 | */ | 527 | */ |
524 | panel = sun4i_tcon_find_panel(node); | 528 | panel = sun4i_tcon_find_panel(node); |
525 | if (IS_ERR(panel)) { | 529 | |
526 | /* | 530 | /* |
527 | * If we don't have a panel endpoint, just go on | 531 | * If we don't have a panel endpoint, just go on |
528 | */ | 532 | */ |
529 | if (PTR_ERR(panel) != -ENODEV) | 533 | if (PTR_ERR(panel) == -EPROBE_DEFER) { |
530 | return -EPROBE_DEFER; | 534 | DRM_DEBUG_DRIVER("Still waiting for our panel. Deferring...\n"); |
535 | return -EPROBE_DEFER; | ||
531 | } | 536 | } |
532 | 537 | ||
533 | return component_add(&pdev->dev, &sun4i_tcon_ops); | 538 | return component_add(&pdev->dev, &sun4i_tcon_ops); |