blob: f7fc6e07ce43df97d61cd2b86bd473033a2608fa [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.felix.dm.itest.api;
import java.util.concurrent.ForkJoinPool;
import java.util.concurrent.TimeUnit;
import org.apache.felix.dm.Component;
import org.apache.felix.dm.ComponentState;
import org.apache.felix.dm.ComponentStateListener;
import org.apache.felix.dm.DependencyManager;
import org.apache.felix.dm.itest.util.Ensure;
import org.apache.felix.dm.itest.util.TestBase;
/**
* Verifies if a concurrent deactivation of two components depending on each other does not produce a deadlock.
*/
public class FELIX5471_CyclicDependencyTest extends TestBase {
volatile Ensure m_ensure;
public void testCyclicDependency() throws InterruptedException {
DependencyManager m = getDM();
ForkJoinPool tpool = new ForkJoinPool(2);
try {
for (int count = 0; count < 1000; count++) {
m_ensure = new Ensure(false);
Component a = m.createComponent()
.setImplementation(new A())
.setInterface(A.class.getName(), null)
.add(m.createServiceDependency().setService(B.class).setRequired(true).setCallbacks("add", "remove"));
Component b = m.createComponent()
.setImplementation(new B())
.setInterface(B.class.getName(), null)
.add(m.createServiceDependency().setService(A.class).setRequired(false).setCallbacks("add", "remove"));
m.add(a);
m.add(b);
ComponentStateListener l = (c, s) -> {
if (s == ComponentState.INACTIVE) {
m_ensure.step();
}
};
a.add(l);
b.add(l);
m_ensure.waitForStep(4, 5000); // A started, B started
tpool.execute(() -> m.remove(a));
tpool.execute(() -> m.remove(b));
m_ensure.waitForStep(10, 5000); // A unbound from B, stopped and inactive, B unbound from A, stopped and inactive
tpool.awaitQuiescence(5000, TimeUnit.MILLISECONDS);
}
} finally {
tpool.shutdown();
}
}
public class Base {
void start() {
m_ensure.step();
}
void stop() {
m_ensure.step();
}
}
public class A extends Base {
public void add(B b) {
m_ensure.step();
}
public void remove(B b) {
m_ensure.step();
}
}
public class B extends Base {
public void add(A a) {
m_ensure.step();
}
public void remove(A a) {
m_ensure.step();
}
}
}