| 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 |