| /* |
| * Copyright (C) 2010 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| // TODO(b/129481165): remove the #pragma below and fix conversion issues |
| #pragma clang diagnostic push |
| #pragma clang diagnostic ignored "-Wconversion" |
| |
| #include <sys/resource.h> |
| |
| #include <sched.h> |
| |
| #include <android/frameworks/displayservice/1.0/IDisplayService.h> |
| #include <android/hardware/configstore/1.0/ISurfaceFlingerConfigs.h> |
| #include <android/hardware/graphics/allocator/2.0/IAllocator.h> |
| #include <android/hardware/graphics/allocator/3.0/IAllocator.h> |
| #include <binder/IPCThreadState.h> |
| #include <binder/IServiceManager.h> |
| #include <binder/ProcessState.h> |
| #include <configstore/Utils.h> |
| #include <displayservice/DisplayService.h> |
| #include <errno.h> |
| #include <hidl/LegacySupport.h> |
| #include <processgroup/sched_policy.h> |
| #include "SurfaceFlinger.h" |
| #include "SurfaceFlingerFactory.h" |
| #include "SurfaceFlingerProperties.h" |
| |
| using namespace android; |
| |
| static status_t startGraphicsAllocatorService() { |
| using android::hardware::configstore::getBool; |
| using android::hardware::configstore::V1_0::ISurfaceFlingerConfigs; |
| if (!android::sysprop::start_graphics_allocator_service(false)) { |
| return OK; |
| } |
| |
| status_t result = hardware::registerPassthroughServiceImplementation< |
| android::hardware::graphics::allocator::V3_0::IAllocator>(); |
| if (result == OK) { |
| return OK; |
| } |
| |
| result = hardware::registerPassthroughServiceImplementation< |
| android::hardware::graphics::allocator::V2_0::IAllocator>(); |
| if (result != OK) { |
| ALOGE("could not start graphics allocator service"); |
| return result; |
| } |
| |
| return OK; |
| } |
| |
| static void startDisplayService() { |
| using android::frameworks::displayservice::V1_0::implementation::DisplayService; |
| using android::frameworks::displayservice::V1_0::IDisplayService; |
| |
| sp<IDisplayService> displayservice = new DisplayService(); |
| status_t err = displayservice->registerAsService(); |
| |
| // b/141930622 |
| if (err != OK) { |
| ALOGE("Did not register (deprecated) IDisplayService service."); |
| } |
| } |
| |
| int main(int, char**) { |
| signal(SIGPIPE, SIG_IGN); |
| |
| hardware::configureRpcThreadpool(1 /* maxThreads */, |
| false /* callerWillJoin */); |
| |
| startGraphicsAllocatorService(); |
| |
| // When SF is launched in its own process, limit the number of |
| // binder threads to 4. |
| ProcessState::self()->setThreadPoolMaxThreadCount(4); |
| |
| // Set uclamp.min setting on all threads, maybe an overkill but we want |
| // to cover important threads like RenderEngine. |
| if (SurfaceFlinger::setSchedAttr(true) != NO_ERROR) { |
| ALOGW("Couldn't set uclamp.min: %s\n", strerror(errno)); |
| } |
| |
| // The binder threadpool we start will inherit sched policy and priority |
| // of (this) creating thread. We want the binder thread pool to have |
| // SCHED_FIFO policy and priority 1 (lowest RT priority) |
| // Once the pool is created we reset this thread's priority back to |
| // original. |
| int newPriority = 0; |
| int origPolicy = sched_getscheduler(0); |
| struct sched_param origSchedParam; |
| |
| int errorInPriorityModification = sched_getparam(0, &origSchedParam); |
| if (errorInPriorityModification == 0) { |
| int policy = SCHED_FIFO; |
| newPriority = sched_get_priority_min(policy); |
| |
| struct sched_param param; |
| param.sched_priority = newPriority; |
| |
| errorInPriorityModification = sched_setscheduler(0, policy, ¶m); |
| } |
| |
| // start the thread pool |
| sp<ProcessState> ps(ProcessState::self()); |
| ps->startThreadPool(); |
| |
| // Reset current thread's policy and priority |
| if (errorInPriorityModification == 0) { |
| errorInPriorityModification = sched_setscheduler(0, origPolicy, &origSchedParam); |
| } else { |
| ALOGE("Failed to set SurfaceFlinger binder threadpool priority to SCHED_FIFO"); |
| } |
| |
| // instantiate surfaceflinger |
| sp<SurfaceFlinger> flinger = surfaceflinger::createSurfaceFlinger(); |
| |
| // Set the minimum policy of surfaceflinger node to be SCHED_FIFO. |
| // So any thread with policy/priority lower than {SCHED_FIFO, 1}, will run |
| // at least with SCHED_FIFO policy and priority 1. |
| if (errorInPriorityModification == 0) { |
| flinger->setMinSchedulerPolicy(SCHED_FIFO, newPriority); |
| } |
| |
| setpriority(PRIO_PROCESS, 0, PRIORITY_URGENT_DISPLAY); |
| |
| set_sched_policy(0, SP_FOREGROUND); |
| |
| // Put most SurfaceFlinger threads in the system-background cpuset |
| // Keeps us from unnecessarily using big cores |
| // Do this after the binder thread pool init |
| if (cpusets_enabled()) set_cpuset_policy(0, SP_SYSTEM); |
| |
| // initialize before clients can connect |
| flinger->init(); |
| |
| // publish surface flinger |
| sp<IServiceManager> sm(defaultServiceManager()); |
| sm->addService(String16(SurfaceFlinger::getServiceName()), flinger, false, |
| IServiceManager::DUMP_FLAG_PRIORITY_CRITICAL | IServiceManager::DUMP_FLAG_PROTO); |
| |
| // publish gui::ISurfaceComposer, the new AIDL interface |
| sp<SurfaceComposerAIDL> composerAIDL = new SurfaceComposerAIDL(flinger); |
| sm->addService(String16("SurfaceFlingerAIDL"), composerAIDL, false, |
| IServiceManager::DUMP_FLAG_PRIORITY_CRITICAL | IServiceManager::DUMP_FLAG_PROTO); |
| |
| startDisplayService(); // dependency on SF getting registered above |
| |
| if (SurfaceFlinger::setSchedFifo(true) != NO_ERROR) { |
| ALOGW("Couldn't set to SCHED_FIFO: %s", strerror(errno)); |
| } |
| |
| // run surface flinger in this thread |
| flinger->run(); |
| |
| return 0; |
| } |
| |
| // TODO(b/129481165): remove the #pragma below and fix conversion issues |
| #pragma clang diagnostic pop // ignored "-Wconversion" |