/** * Copyright 2014 Netflix, Inc. * * 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. */ package rx.schedulers; import rx.Scheduler; import rx.Subscription; import rx.functions.Action0; import rx.internal.schedulers.NewThreadWorker; import rx.internal.schedulers.ScheduledAction; import rx.internal.util.RxThreadFactory; import rx.subscriptions.CompositeSubscription; import rx.subscriptions.Subscriptions; import java.util.concurrent.ThreadFactory; import java.util.concurrent.TimeUnit; /* package */class EventLoopsScheduler extends Scheduler { /** Manages a fixed number of workers. */ private static final String THREAD_NAME_PREFIX = "RxComputationThreadPool-"; private static final RxThreadFactory THREAD_FACTORY = new RxThreadFactory(THREAD_NAME_PREFIX); static final class FixedSchedulerPool { final int cores; final PoolWorker[] eventLoops; long n; FixedSchedulerPool() { // initialize event loops this.cores = Runtime.getRuntime().availableProcessors(); this.eventLoops = new PoolWorker[cores]; for (int i = 0; i < cores; i++) { this.eventLoops[i] = new PoolWorker(THREAD_FACTORY); } } public PoolWorker getEventLoop() { // simple round robin, improvements to come return eventLoops[(int)(n++ % cores)]; } } final FixedSchedulerPool pool; /** * Create a scheduler with pool size equal to the available processor * count and using least-recent worker selection policy. */ EventLoopsScheduler() { pool = new FixedSchedulerPool(); } @Override public Worker createWorker() { return new EventLoopWorker(pool.getEventLoop()); } private static class EventLoopWorker extends Scheduler.Worker { private final CompositeSubscription innerSubscription = new CompositeSubscription(); private final PoolWorker poolWorker; EventLoopWorker(PoolWorker poolWorker) { this.poolWorker = poolWorker; } @Override public void unsubscribe() { innerSubscription.unsubscribe(); } @Override public boolean isUnsubscribed() { return innerSubscription.isUnsubscribed(); } @Override public Subscription schedule(Action0 action) { return schedule(action, 0, null); } @Override public Subscription schedule(Action0 action, long delayTime, TimeUnit unit) { if (innerSubscription.isUnsubscribed()) { // don't schedule, we are unsubscribed return Subscriptions.empty(); } ScheduledAction s = poolWorker.scheduleActual(action, delayTime, unit); innerSubscription.add(s); s.addParent(innerSubscription); return s; } } private static final class PoolWorker extends NewThreadWorker { PoolWorker(ThreadFactory threadFactory) { super(threadFactory); } } }