OLD | NEW |
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "ui/gfx/ozone/impl/software_surface_factory_ozone.h" | 5 #include "ui/gfx/ozone/impl/dri_surface_factory.h" |
6 | 6 |
7 #include <drm.h> | 7 #include <drm.h> |
8 #include <errno.h> | 8 #include <errno.h> |
9 #include <xf86drm.h> | 9 #include <xf86drm.h> |
10 | 10 |
11 #include "base/message_loop/message_loop.h" | 11 #include "base/message_loop/message_loop.h" |
12 #include "third_party/skia/include/core/SkBitmap.h" | 12 #include "third_party/skia/include/core/SkBitmap.h" |
13 #include "third_party/skia/include/core/SkDevice.h" | 13 #include "third_party/skia/include/core/SkDevice.h" |
14 #include "ui/gfx/native_widget_types.h" | 14 #include "ui/gfx/native_widget_types.h" |
15 #include "ui/gfx/ozone/impl/drm_skbitmap_ozone.h" | 15 #include "ui/gfx/ozone/impl/dri_skbitmap.h" |
16 #include "ui/gfx/ozone/impl/drm_wrapper_ozone.h" | 16 #include "ui/gfx/ozone/impl/dri_surface.h" |
17 #include "ui/gfx/ozone/impl/hardware_display_controller_ozone.h" | 17 #include "ui/gfx/ozone/impl/dri_wrapper.h" |
18 #include "ui/gfx/ozone/impl/software_surface_ozone.h" | 18 #include "ui/gfx/ozone/impl/hardware_display_controller.h" |
19 | 19 |
20 namespace gfx { | 20 namespace gfx { |
21 | 21 |
22 namespace { | 22 namespace { |
23 | 23 |
24 const char kDefaultGraphicsCardPath[] = "/dev/dri/card0"; | 24 const char kDefaultGraphicsCardPath[] = "/dev/dri/card0"; |
25 const char kDPMSProperty[] = "DPMS"; | 25 const char kDPMSProperty[] = "DPMS"; |
26 | 26 |
27 const gfx::AcceleratedWidget kDefaultWidgetHandle = 1; | 27 const gfx::AcceleratedWidget kDefaultWidgetHandle = 1; |
28 | 28 |
29 // DRM callback on page flip events. This callback is triggered after the | 29 // DRM callback on page flip events. This callback is triggered after the |
30 // page flip has happened and the backbuffer is now the new frontbuffer | 30 // page flip has happened and the backbuffer is now the new frontbuffer |
31 // The old frontbuffer is no longer used by the hardware and can be used for | 31 // The old frontbuffer is no longer used by the hardware and can be used for |
32 // future draw operations. | 32 // future draw operations. |
33 // | 33 // |
34 // |device| will contain a reference to the |SoftwareSurfaceOzone| object which | 34 // |device| will contain a reference to the |DriSurface| object which |
35 // the event belongs to. | 35 // the event belongs to. |
36 // | 36 // |
37 // TODO(dnicoara) When we have a FD handler for the DRM calls in the message | 37 // TODO(dnicoara) When we have a FD handler for the DRM calls in the message |
38 // loop, we can move this function in the handler. | 38 // loop, we can move this function in the handler. |
39 void HandlePageFlipEvent(int fd, | 39 void HandlePageFlipEvent(int fd, |
40 unsigned int frame, | 40 unsigned int frame, |
41 unsigned int seconds, | 41 unsigned int seconds, |
42 unsigned int useconds, | 42 unsigned int useconds, |
43 void* controller) { | 43 void* controller) { |
44 static_cast<HardwareDisplayControllerOzone*>(controller)->get_surface() | 44 static_cast<HardwareDisplayController*>(controller)->get_surface() |
45 ->SwapBuffers(); | 45 ->SwapBuffers(); |
46 } | 46 } |
47 | 47 |
48 uint32_t GetDrmProperty(int fd, drmModeConnector* connector, const char* name) { | 48 uint32_t GetDriProperty(int fd, drmModeConnector* connector, const char* name) { |
49 for (int i = 0; i < connector->count_props; ++i) { | 49 for (int i = 0; i < connector->count_props; ++i) { |
50 drmModePropertyPtr property = drmModeGetProperty(fd, connector->props[i]); | 50 drmModePropertyPtr property = drmModeGetProperty(fd, connector->props[i]); |
51 if (!property) | 51 if (!property) |
52 continue; | 52 continue; |
53 | 53 |
54 if (strcmp(property->name, name) == 0) { | 54 if (strcmp(property->name, name) == 0) { |
55 uint32_t id = property->prop_id; | 55 uint32_t id = property->prop_id; |
56 drmModeFreeProperty(property); | 56 drmModeFreeProperty(property); |
57 return id; | 57 return id; |
58 } | 58 } |
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
90 drmModeFreeEncoder(encoder); | 90 drmModeFreeEncoder(encoder); |
91 return resources->crtcs[j]; | 91 return resources->crtcs[j]; |
92 } | 92 } |
93 } | 93 } |
94 | 94 |
95 return 0; | 95 return 0; |
96 } | 96 } |
97 | 97 |
98 } // namespace | 98 } // namespace |
99 | 99 |
100 SoftwareSurfaceFactoryOzone::SoftwareSurfaceFactoryOzone() | 100 DriSurfaceFactory::DriSurfaceFactory() |
101 : drm_(), | 101 : drm_(), |
102 state_(UNINITIALIZED), | 102 state_(UNINITIALIZED), |
103 controller_() { | 103 controller_() { |
104 } | 104 } |
105 | 105 |
106 SoftwareSurfaceFactoryOzone::~SoftwareSurfaceFactoryOzone() { | 106 DriSurfaceFactory::~DriSurfaceFactory() { |
107 if (state_ == INITIALIZED) | 107 if (state_ == INITIALIZED) |
108 ShutdownHardware(); | 108 ShutdownHardware(); |
109 } | 109 } |
110 | 110 |
111 SurfaceFactoryOzone::HardwareState | 111 SurfaceFactoryOzone::HardwareState |
112 SoftwareSurfaceFactoryOzone::InitializeHardware() { | 112 DriSurfaceFactory::InitializeHardware() { |
113 CHECK(state_ == UNINITIALIZED); | 113 CHECK(state_ == UNINITIALIZED); |
114 | 114 |
115 // TODO(dnicoara): Short-cut right now. What we want is to look at all the | 115 // TODO(dnicoara): Short-cut right now. What we want is to look at all the |
116 // graphics devices available and select the primary one. | 116 // graphics devices available and select the primary one. |
117 drm_.reset(CreateWrapper()); | 117 drm_.reset(CreateWrapper()); |
118 if (drm_->get_fd() < 0) { | 118 if (drm_->get_fd() < 0) { |
119 LOG(ERROR) << "Cannot open graphics card '" | 119 LOG(ERROR) << "Cannot open graphics card '" |
120 << kDefaultGraphicsCardPath << "': " << strerror(errno); | 120 << kDefaultGraphicsCardPath << "': " << strerror(errno); |
121 state_ = FAILED; | 121 state_ = FAILED; |
122 return state_; | 122 return state_; |
123 } | 123 } |
124 | 124 |
125 state_ = INITIALIZED; | 125 state_ = INITIALIZED; |
126 return state_; | 126 return state_; |
127 } | 127 } |
128 | 128 |
129 void SoftwareSurfaceFactoryOzone::ShutdownHardware() { | 129 void DriSurfaceFactory::ShutdownHardware() { |
130 CHECK(state_ == INITIALIZED); | 130 CHECK(state_ == INITIALIZED); |
131 | 131 |
132 controller_.reset(); | 132 controller_.reset(); |
133 drm_.reset(); | 133 drm_.reset(); |
134 | 134 |
135 state_ = UNINITIALIZED; | 135 state_ = UNINITIALIZED; |
136 } | 136 } |
137 | 137 |
138 gfx::AcceleratedWidget SoftwareSurfaceFactoryOzone::GetAcceleratedWidget() { | 138 gfx::AcceleratedWidget DriSurfaceFactory::GetAcceleratedWidget() { |
139 CHECK(state_ != FAILED); | 139 CHECK(state_ != FAILED); |
140 | 140 |
141 // TODO(dnicoara) When there's more information on which display we want, | 141 // TODO(dnicoara) When there's more information on which display we want, |
142 // then we can return the widget associated with the display. | 142 // then we can return the widget associated with the display. |
143 // For now just assume we have 1 display device and return it. | 143 // For now just assume we have 1 display device and return it. |
144 if (!controller_.get()) | 144 if (!controller_.get()) |
145 controller_.reset(new HardwareDisplayControllerOzone()); | 145 controller_.reset(new HardwareDisplayController()); |
146 | 146 |
147 // TODO(dnicoara) We only have 1 display for now, so only 1 AcceleratedWidget. | 147 // TODO(dnicoara) We only have 1 display for now, so only 1 AcceleratedWidget. |
148 // When we'll support multiple displays this needs to be changed to return a | 148 // When we'll support multiple displays this needs to be changed to return a |
149 // different handle for every display. | 149 // different handle for every display. |
150 return kDefaultWidgetHandle; | 150 return kDefaultWidgetHandle; |
151 } | 151 } |
152 | 152 |
153 gfx::AcceleratedWidget SoftwareSurfaceFactoryOzone::RealizeAcceleratedWidget( | 153 gfx::AcceleratedWidget DriSurfaceFactory::RealizeAcceleratedWidget( |
154 gfx::AcceleratedWidget w) { | 154 gfx::AcceleratedWidget w) { |
155 CHECK(state_ == INITIALIZED); | 155 CHECK(state_ == INITIALIZED); |
156 // TODO(dnicoara) Once we can handle multiple displays this needs to be | 156 // TODO(dnicoara) Once we can handle multiple displays this needs to be |
157 // changed. | 157 // changed. |
158 CHECK(w == kDefaultWidgetHandle); | 158 CHECK(w == kDefaultWidgetHandle); |
159 | 159 |
160 CHECK(controller_->get_state() == | 160 CHECK(controller_->get_state() == |
161 HardwareDisplayControllerOzone::UNASSOCIATED); | 161 HardwareDisplayController::UNASSOCIATED); |
162 | 162 |
163 // Until now the controller is just a stub. Initializing it will link it to a | 163 // Until now the controller is just a stub. Initializing it will link it to a |
164 // hardware display. | 164 // hardware display. |
165 if (!InitializeControllerForPrimaryDisplay(drm_.get(), controller_.get())) { | 165 if (!InitializeControllerForPrimaryDisplay(drm_.get(), controller_.get())) { |
166 LOG(ERROR) << "Failed to initialize controller"; | 166 LOG(ERROR) << "Failed to initialize controller"; |
167 return gfx::kNullAcceleratedWidget; | 167 return gfx::kNullAcceleratedWidget; |
168 } | 168 } |
169 | 169 |
170 // Create a surface suitable for the current controller. | 170 // Create a surface suitable for the current controller. |
171 scoped_ptr<SoftwareSurfaceOzone> surface(CreateSurface(controller_.get())); | 171 scoped_ptr<DriSurface> surface(CreateSurface(controller_.get())); |
172 | 172 |
173 if (!surface->Initialize()) { | 173 if (!surface->Initialize()) { |
174 LOG(ERROR) << "Failed to initialize surface"; | 174 LOG(ERROR) << "Failed to initialize surface"; |
175 return gfx::kNullAcceleratedWidget; | 175 return gfx::kNullAcceleratedWidget; |
176 } | 176 } |
177 | 177 |
178 // Bind the surface to the controller. This will register the backing buffers | 178 // Bind the surface to the controller. This will register the backing buffers |
179 // with the hardware CRTC such that we can show the buffers. The controller | 179 // with the hardware CRTC such that we can show the buffers. The controller |
180 // takes ownership of the surface. | 180 // takes ownership of the surface. |
181 if (!controller_->BindSurfaceToController(surface.Pass())) { | 181 if (!controller_->BindSurfaceToController(surface.Pass())) { |
182 LOG(ERROR) << "Failed to bind surface to controller"; | 182 LOG(ERROR) << "Failed to bind surface to controller"; |
183 return gfx::kNullAcceleratedWidget; | 183 return gfx::kNullAcceleratedWidget; |
184 } | 184 } |
185 | 185 |
186 return reinterpret_cast<gfx::AcceleratedWidget>(controller_->get_surface()); | 186 return reinterpret_cast<gfx::AcceleratedWidget>(controller_->get_surface()); |
187 } | 187 } |
188 | 188 |
189 bool SoftwareSurfaceFactoryOzone::LoadEGLGLES2Bindings( | 189 bool DriSurfaceFactory::LoadEGLGLES2Bindings( |
190 AddGLLibraryCallback add_gl_library, | 190 AddGLLibraryCallback add_gl_library, |
191 SetGLGetProcAddressProcCallback set_gl_get_proc_address) { | 191 SetGLGetProcAddressProcCallback set_gl_get_proc_address) { |
192 return false; | 192 return false; |
193 } | 193 } |
194 | 194 |
195 bool SoftwareSurfaceFactoryOzone::AttemptToResizeAcceleratedWidget( | 195 bool DriSurfaceFactory::AttemptToResizeAcceleratedWidget( |
196 gfx::AcceleratedWidget w, | 196 gfx::AcceleratedWidget w, |
197 const gfx::Rect& bounds) { | 197 const gfx::Rect& bounds) { |
198 return false; | 198 return false; |
199 } | 199 } |
200 | 200 |
201 bool SoftwareSurfaceFactoryOzone::SchedulePageFlip(gfx::AcceleratedWidget w) { | 201 bool DriSurfaceFactory::SchedulePageFlip(gfx::AcceleratedWidget w) { |
202 CHECK(state_ == INITIALIZED); | 202 CHECK(state_ == INITIALIZED); |
203 // TODO(dnicoara) Change this CHECK once we're running with the threaded | 203 // TODO(dnicoara) Change this CHECK once we're running with the threaded |
204 // compositor. | 204 // compositor. |
205 CHECK(base::MessageLoop::current()->type() == base::MessageLoop::TYPE_UI); | 205 CHECK(base::MessageLoop::current()->type() == base::MessageLoop::TYPE_UI); |
206 | 206 |
207 // TODO(dnicoara) Once we can handle multiple displays this needs to be | 207 // TODO(dnicoara) Once we can handle multiple displays this needs to be |
208 // changed. | 208 // changed. |
209 CHECK(w == kDefaultWidgetHandle); | 209 CHECK(w == kDefaultWidgetHandle); |
210 | 210 |
211 if (!controller_->SchedulePageFlip()) | 211 if (!controller_->SchedulePageFlip()) |
212 return false; | 212 return false; |
213 | 213 |
214 // Only wait for the page flip event to finish if it was properly scheduled. | 214 // Only wait for the page flip event to finish if it was properly scheduled. |
215 // | 215 // |
216 // TODO(dnicoara) The following call will wait for the page flip event to | 216 // TODO(dnicoara) The following call will wait for the page flip event to |
217 // complete. This means that it will block until the next VSync. Ideally the | 217 // complete. This means that it will block until the next VSync. Ideally the |
218 // wait should happen in the message loop. The message loop would then | 218 // wait should happen in the message loop. The message loop would then |
219 // schedule the next draw event. Alternatively, the VSyncProvider could be | 219 // schedule the next draw event. Alternatively, the VSyncProvider could be |
220 // used to schedule the next draw. Unfortunately, at this point, | 220 // used to schedule the next draw. Unfortunately, at this point, |
221 // SoftwareOutputDevice does not provide any means to use any of the above | 221 // DriOutputDevice does not provide any means to use any of the above |
222 // solutions. Note that if the DRM callback does not schedule the next draw, | 222 // solutions. Note that if the DRM callback does not schedule the next draw, |
223 // then some sort of synchronization needs to take place since starting a new | 223 // then some sort of synchronization needs to take place since starting a new |
224 // draw before the page flip happened is considered an error. However we can | 224 // draw before the page flip happened is considered an error. However we can |
225 // not use any lock constructs unless we're using the threaded compositor. | 225 // not use any lock constructs unless we're using the threaded compositor. |
226 // Note that the following call does not use any locks, so it is safe to be | 226 // Note that the following call does not use any locks, so it is safe to be |
227 // made on the UI thread (thought not ideal). | 227 // made on the UI thread (thought not ideal). |
228 WaitForPageFlipEvent(drm_->get_fd()); | 228 WaitForPageFlipEvent(drm_->get_fd()); |
229 | 229 |
230 return true; | 230 return true; |
231 } | 231 } |
232 | 232 |
233 SkCanvas* SoftwareSurfaceFactoryOzone::GetCanvasForWidget( | 233 SkCanvas* DriSurfaceFactory::GetCanvasForWidget( |
234 gfx::AcceleratedWidget w) { | 234 gfx::AcceleratedWidget w) { |
235 CHECK(state_ == INITIALIZED); | 235 CHECK(state_ == INITIALIZED); |
236 return reinterpret_cast<SoftwareSurfaceOzone*>(w)->GetDrawableForWidget(); | 236 return reinterpret_cast<DriSurface*>(w)->GetDrawableForWidget(); |
237 } | 237 } |
238 | 238 |
239 gfx::VSyncProvider* SoftwareSurfaceFactoryOzone::GetVSyncProvider( | 239 gfx::VSyncProvider* DriSurfaceFactory::GetVSyncProvider( |
240 gfx::AcceleratedWidget w) { | 240 gfx::AcceleratedWidget w) { |
241 return NULL; | 241 return NULL; |
242 } | 242 } |
243 | 243 |
244 //////////////////////////////////////////////////////////////////////////////// | 244 //////////////////////////////////////////////////////////////////////////////// |
245 // SoftwareSurfaceFactoryOzone private | 245 // DriSurfaceFactory private |
246 | 246 |
247 SoftwareSurfaceOzone* SoftwareSurfaceFactoryOzone::CreateSurface( | 247 DriSurface* DriSurfaceFactory::CreateSurface( |
248 HardwareDisplayControllerOzone* controller) { | 248 HardwareDisplayController* controller) { |
249 return new SoftwareSurfaceOzone(controller); | 249 return new DriSurface(controller); |
250 } | 250 } |
251 | 251 |
252 DrmWrapperOzone* SoftwareSurfaceFactoryOzone::CreateWrapper() { | 252 DriWrapper* DriSurfaceFactory::CreateWrapper() { |
253 return new DrmWrapperOzone(kDefaultGraphicsCardPath); | 253 return new DriWrapper(kDefaultGraphicsCardPath); |
254 } | 254 } |
255 | 255 |
256 bool SoftwareSurfaceFactoryOzone::InitializeControllerForPrimaryDisplay( | 256 bool DriSurfaceFactory::InitializeControllerForPrimaryDisplay( |
257 DrmWrapperOzone* drm, | 257 DriWrapper* drm, |
258 HardwareDisplayControllerOzone* controller) { | 258 HardwareDisplayController* controller) { |
259 CHECK(state_ == SurfaceFactoryOzone::INITIALIZED); | 259 CHECK(state_ == SurfaceFactoryOzone::INITIALIZED); |
260 | 260 |
261 drmModeRes* resources = drmModeGetResources(drm->get_fd()); | 261 drmModeRes* resources = drmModeGetResources(drm->get_fd()); |
262 | 262 |
263 // Search for an active connector. | 263 // Search for an active connector. |
264 for (int i = 0; i < resources->count_connectors; ++i) { | 264 for (int i = 0; i < resources->count_connectors; ++i) { |
265 drmModeConnector* connector = drmModeGetConnector( | 265 drmModeConnector* connector = drmModeGetConnector( |
266 drm->get_fd(), | 266 drm->get_fd(), |
267 resources->connectors[i]); | 267 resources->connectors[i]); |
268 | 268 |
269 if (!connector) | 269 if (!connector) |
270 continue; | 270 continue; |
271 | 271 |
272 if (connector->connection != DRM_MODE_CONNECTED || | 272 if (connector->connection != DRM_MODE_CONNECTED || |
273 connector->count_modes == 0) { | 273 connector->count_modes == 0) { |
274 drmModeFreeConnector(connector); | 274 drmModeFreeConnector(connector); |
275 continue; | 275 continue; |
276 } | 276 } |
277 | 277 |
278 uint32_t crtc = GetCrtc(drm->get_fd(), resources, connector); | 278 uint32_t crtc = GetCrtc(drm->get_fd(), resources, connector); |
279 | 279 |
280 if (!crtc) | 280 if (!crtc) |
281 continue; | 281 continue; |
282 | 282 |
283 uint32_t dpms_property_id = GetDrmProperty(drm->get_fd(), | 283 uint32_t dpms_property_id = GetDriProperty(drm->get_fd(), |
284 connector, | 284 connector, |
285 kDPMSProperty); | 285 kDPMSProperty); |
286 | 286 |
287 // TODO(dnicoara) Select one mode for now. In the future we may need to | 287 // TODO(dnicoara) Select one mode for now. In the future we may need to |
288 // save all the modes and allow the user to choose a specific mode. Or | 288 // save all the modes and allow the user to choose a specific mode. Or |
289 // even some fullscreen applications may need to change the mode. | 289 // even some fullscreen applications may need to change the mode. |
290 controller->SetControllerInfo( | 290 controller->SetControllerInfo( |
291 drm, | 291 drm, |
292 connector->connector_id, | 292 connector->connector_id, |
293 crtc, | 293 crtc, |
294 dpms_property_id, | 294 dpms_property_id, |
295 connector->modes[0]); | 295 connector->modes[0]); |
296 | 296 |
297 drmModeFreeConnector(connector); | 297 drmModeFreeConnector(connector); |
298 | 298 |
299 return true; | 299 return true; |
300 } | 300 } |
301 | 301 |
302 return false; | 302 return false; |
303 } | 303 } |
304 | 304 |
305 void SoftwareSurfaceFactoryOzone::WaitForPageFlipEvent(int fd) { | 305 void DriSurfaceFactory::WaitForPageFlipEvent(int fd) { |
306 drmEventContext drm_event; | 306 drmEventContext drm_event; |
307 drm_event.version = DRM_EVENT_CONTEXT_VERSION; | 307 drm_event.version = DRM_EVENT_CONTEXT_VERSION; |
308 drm_event.page_flip_handler = HandlePageFlipEvent; | 308 drm_event.page_flip_handler = HandlePageFlipEvent; |
309 drm_event.vblank_handler = NULL; | 309 drm_event.vblank_handler = NULL; |
310 | 310 |
311 // Wait for the page-flip to complete. | 311 // Wait for the page-flip to complete. |
312 drmHandleEvent(fd, &drm_event); | 312 drmHandleEvent(fd, &drm_event); |
313 } | 313 } |
314 | 314 |
315 } // namespace gfx | 315 } // namespace gfx |
OLD | NEW |