All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jclarion.clarion.ClarionMemoryModel Maven / Gradle / Ivy

/**
 * Copyright 2010, by Andrew Barnham
 *
 * The contents of this file are subject to
 * GNU Lesser General Public License (LGPL), v.3
 * http://www.gnu.org/licenses/lgpl.txt
 * 
 * Software distributed under the License is distributed on an "AS IS" basis,
 * WITHOUT WARRANTY OF ANY KIND, either express or implied.
 */
package org.jclarion.clarion;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

import org.jclarion.clarion.memory.CMem;
import org.jclarion.clarion.primative.AbstractStateFactory;
import org.jclarion.clarion.primative.AbstractStateGetter;
import org.jclarion.clarion.primative.GlobalStateGetter;
import org.jclarion.clarion.primative.MyReference;
import org.jclarion.clarion.primative.MyStrongReference;
import org.jclarion.clarion.primative.MyWeakReference;
import org.jclarion.clarion.primative.ThreadStateGetter;
import org.jclarion.clarion.runtime.CMemory;
import org.jclarion.clarion.runtime.OverGlue;


/**
 * Model clarion object on memory.
 * 
 * @author barney
 *
 */
public abstract class ClarionMemoryModel implements Threaded 
{
    public static class State
    {
        public State(State base) {
            listeners=base.listeners;
            base.listeners=null;
        }

        public State() {
        }

        private List listeners;
    }
    
    protected void lock(ClarionMemoryModel base,Thread t)
    {
        state=base.state.getLockedGetter(t);
    }
 
    public ClarionMemoryModel castTo(Class clazz)
    {
        try {
            ClarionMemoryModel model = clazz.newInstance();
            if (model instanceof ClarionGroup) {
                ClarionGroup cg = (ClarionGroup)model;
                int cglen = CMemory.size(cg);
                int thislen = CMemory.size(this);
                if (thislen>cglen) {
                    cg.addVariable("__padding:"+cg.getVariableCount(),new ClarionString(thislen-cglen));
                }
            }
            (new OverGlue(model,this)).objectChanged(this);
            return model;
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
        return null;
    }
    
    private static class StateFactory extends AbstractStateFactory
    {
        @Override
        public State cloneState(State base) {
            return new State(base);
        }

        @Override
        public State createState() {
            return new State();
        }
    }
    private static StateFactory factory=new StateFactory();
    
    private AbstractStateGetter state;
    
    public ClarionMemoryModel() {
        state=new GlobalStateGetter(factory);
    }

    public ClarionMemoryModel(ClarionMemoryModel base,Thread lock) {
        if (lock!=null) {
            state=base.state.getLockedGetter(lock);
        } else {
            state=new GlobalStateGetter(factory);
        }
    }

    
    public void initThread()
    {
        if (!state.isThreaded()) {
            state=new ThreadStateGetter(factory,state);
            state.get();
        }
    }

    /**
     * Write java object to memory as clarion would represent object in memory
     * 
     * @param is
     */
    public abstract void serialize(CMem is);
    
    /**
     * Read object from memory as clarion would represent object in memory
     */ 
    public abstract void deserialize(CMem os);
    
    /**
     * Listen for changes to the objects representation. Note that listener
     * is weakly referenced. You need to maintain a strong reference to the
     * listener object in order for it not to be garbage collected 
     * 
     * @param cmcl
     */
    public void addChangeListener(ClarionMemoryChangeListener cmcl)
    {
        addChangeListener(cmcl,false);
    }

    public void addChangeListener(ClarionMemoryChangeListener cmcl,boolean strong)
    {
        State s = state.get();
        synchronized(s) {
            if (s.listeners==null) {
                s.listeners = new ArrayList();
            }
            if (strong) {
                s.listeners.add(new MyStrongReference(cmcl));
            } else {
                s.listeners.add(new MyWeakReference(cmcl));
            }
        }
    }

    /**
     * Stop Listening for changes to the objects representation
     * 
     * @param cmcl
     */
    public void removeChangeListener(ClarionMemoryChangeListener cmcl)
    {
        State s = state.get();
        synchronized(s) {
            if (s.listeners==null) return;
            Iterator scan;
            scan = s.listeners.iterator();
            while (scan.hasNext()) {
                MyReference val = scan.next();
                ClarionMemoryChangeListener sval = val.get();
                if (sval==cmcl || sval==null) {
                    scan.remove();
                }
            }
            if (s.listeners.isEmpty()) s.listeners=null;
        }
    }
    
    public boolean isAnyoneInterestedInChange()
    {
        State s = state.get();
        synchronized(s) {
            return s.listeners!=null;
        }
    }
    
    public int getListenerCount()
    {
        State s = state.get();
        synchronized(s) {
            if (s.listeners==null) return 0;
            return s.listeners.size();
        }
    }
    
    public void notifyChange() {

        List notify = null; 
        State s = state.get();
        synchronized(s) {
            if (s.listeners==null) return;
            Iterator scan;
            scan = s.listeners.iterator();
            while (scan.hasNext()) {
                MyReference val = scan.next();
                ClarionMemoryChangeListener sval = val.get();
                if (sval==null) {
                    scan.remove();
                } else {
                    if (notify==null) {
                        notify = new ArrayList();                        
                    }
                    notify.add(sval);
                }
            }
            if (s.listeners.isEmpty()) s.listeners=null;
        }
        
        if (notify==null) return;
        
        for ( ClarionMemoryChangeListener scan : notify ) {
            scan.objectChanged(this);
        }
    }
    
    @Override
    public final boolean isThreaded() {
        return state.isThreaded();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy