xref: /AOO41X/main/jurt/com/sun/star/lib/util/AsynchronousFinalizer.java (revision cdf0e10c4e3984b49a9502b011690b615761d4a3)
1*cdf0e10cSrcweir /*************************************************************************
2*cdf0e10cSrcweir  *
3*cdf0e10cSrcweir  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4*cdf0e10cSrcweir  *
5*cdf0e10cSrcweir  * Copyright 2000, 2010 Oracle and/or its affiliates.
6*cdf0e10cSrcweir  *
7*cdf0e10cSrcweir  * OpenOffice.org - a multi-platform office productivity suite
8*cdf0e10cSrcweir  *
9*cdf0e10cSrcweir  * This file is part of OpenOffice.org.
10*cdf0e10cSrcweir  *
11*cdf0e10cSrcweir  * OpenOffice.org is free software: you can redistribute it and/or modify
12*cdf0e10cSrcweir  * it under the terms of the GNU Lesser General Public License version 3
13*cdf0e10cSrcweir  * only, as published by the Free Software Foundation.
14*cdf0e10cSrcweir  *
15*cdf0e10cSrcweir  * OpenOffice.org is distributed in the hope that it will be useful,
16*cdf0e10cSrcweir  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17*cdf0e10cSrcweir  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18*cdf0e10cSrcweir  * GNU Lesser General Public License version 3 for more details
19*cdf0e10cSrcweir  * (a copy is included in the LICENSE file that accompanied this code).
20*cdf0e10cSrcweir  *
21*cdf0e10cSrcweir  * You should have received a copy of the GNU Lesser General Public License
22*cdf0e10cSrcweir  * version 3 along with OpenOffice.org.  If not, see
23*cdf0e10cSrcweir  * <http://www.openoffice.org/license.html>
24*cdf0e10cSrcweir  * for a copy of the LGPLv3 License.
25*cdf0e10cSrcweir  *
26*cdf0e10cSrcweir  ************************************************************************/
27*cdf0e10cSrcweir 
28*cdf0e10cSrcweir package com.sun.star.lib.util;
29*cdf0e10cSrcweir 
30*cdf0e10cSrcweir import java.util.LinkedList;
31*cdf0e10cSrcweir 
32*cdf0e10cSrcweir /**
33*cdf0e10cSrcweir    Helper class to asynchronously execute finalize methods.
34*cdf0e10cSrcweir 
35*cdf0e10cSrcweir    Current JVMs seem not to be robust against long-running finalize methods, in
36*cdf0e10cSrcweir    that such long-running finalize methods may lead to OutOfMemoryErrors.  This
37*cdf0e10cSrcweir    class mitigates the problem by asynchronously shifting the bodies of
38*cdf0e10cSrcweir    potentially long-running finalize methods into an extra thread.  Classes that
39*cdf0e10cSrcweir    make use of this in their finalize methods are the proxies used in the
40*cdf0e10cSrcweir    intra-process JNI UNO bridge and the inter-process Java URP UNO bridge (where
41*cdf0e10cSrcweir    in both cases finalizers lead to synchronous UNO release calls).
42*cdf0e10cSrcweir 
43*cdf0e10cSrcweir    If JVMs are getting more mature and should no longer have problems with
44*cdf0e10cSrcweir    long-running finalize mehtods, this class could be removed again.
45*cdf0e10cSrcweir */
46*cdf0e10cSrcweir public final class AsynchronousFinalizer {
47*cdf0e10cSrcweir     /**
48*cdf0e10cSrcweir        Add a job to be executed asynchronously.
49*cdf0e10cSrcweir 
50*cdf0e10cSrcweir        The run method of the given job is called exactly once.  If it terminates
51*cdf0e10cSrcweir        abnormally by throwing any Throwable, that is ignored.
52*cdf0e10cSrcweir 
53*cdf0e10cSrcweir        @param job represents the body of some finalize method; must not be null.
54*cdf0e10cSrcweir     */
55*cdf0e10cSrcweir     public static void add(Job job) {
56*cdf0e10cSrcweir         synchronized (queue) {
57*cdf0e10cSrcweir             boolean first = queue.isEmpty();
58*cdf0e10cSrcweir             queue.add(job);
59*cdf0e10cSrcweir             if (first) {
60*cdf0e10cSrcweir                 queue.notify();
61*cdf0e10cSrcweir             }
62*cdf0e10cSrcweir         }
63*cdf0e10cSrcweir     }
64*cdf0e10cSrcweir 
65*cdf0e10cSrcweir     /**
66*cdf0e10cSrcweir        An interface to represent bodies of finalize methods.
67*cdf0e10cSrcweir 
68*cdf0e10cSrcweir        Similar to Runnable, except that the run method may throw any Throwable
69*cdf0e10cSrcweir        (which is effectively ignored by AsynchronousFinalizer.add, similar to
70*cdf0e10cSrcweir        any Throwables raised by finalize being ignored).
71*cdf0e10cSrcweir     */
72*cdf0e10cSrcweir     public interface Job {
73*cdf0e10cSrcweir         void run() throws Throwable;
74*cdf0e10cSrcweir     }
75*cdf0e10cSrcweir 
76*cdf0e10cSrcweir     private static final LinkedList queue = new LinkedList();
77*cdf0e10cSrcweir 
78*cdf0e10cSrcweir     static {
79*cdf0e10cSrcweir         Thread t = new Thread() {
80*cdf0e10cSrcweir                 public void run() {
81*cdf0e10cSrcweir                     for (;;) {
82*cdf0e10cSrcweir                         Job j;
83*cdf0e10cSrcweir                         synchronized (queue) {
84*cdf0e10cSrcweir                             while (queue.isEmpty()) {
85*cdf0e10cSrcweir                                 try {
86*cdf0e10cSrcweir                                     queue.wait();
87*cdf0e10cSrcweir                                 } catch (InterruptedException e) {}
88*cdf0e10cSrcweir                             }
89*cdf0e10cSrcweir                             j = (Job) queue.remove(0);
90*cdf0e10cSrcweir                         }
91*cdf0e10cSrcweir                         try {
92*cdf0e10cSrcweir                             j.run();
93*cdf0e10cSrcweir                         } catch (Throwable e) {}
94*cdf0e10cSrcweir                     }
95*cdf0e10cSrcweir                 }
96*cdf0e10cSrcweir             };
97*cdf0e10cSrcweir         t.setDaemon(true);
98*cdf0e10cSrcweir         t.start();
99*cdf0e10cSrcweir     }
100*cdf0e10cSrcweir 
101*cdf0e10cSrcweir     private AsynchronousFinalizer() {}
102*cdf0e10cSrcweir }
103