coffee manager mote interface for sky motes
This commit is contained in:
parent
8fcff235e2
commit
cdcf783ac6
|
@ -26,7 +26,7 @@
|
||||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||||
* SUCH DAMAGE.
|
* SUCH DAMAGE.
|
||||||
*
|
*
|
||||||
* $Id: SkyMoteType.java,v 1.9 2009/04/20 16:12:01 fros4943 Exp $
|
* $Id: SkyMoteType.java,v 1.10 2009/08/11 17:09:34 fros4943 Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package se.sics.cooja.mspmote;
|
package se.sics.cooja.mspmote;
|
||||||
|
@ -49,6 +49,7 @@ import se.sics.cooja.mspmote.interfaces.MspClock;
|
||||||
import se.sics.cooja.mspmote.interfaces.MspMoteID;
|
import se.sics.cooja.mspmote.interfaces.MspMoteID;
|
||||||
import se.sics.cooja.mspmote.interfaces.SkyButton;
|
import se.sics.cooja.mspmote.interfaces.SkyButton;
|
||||||
import se.sics.cooja.mspmote.interfaces.SkyByteRadio;
|
import se.sics.cooja.mspmote.interfaces.SkyByteRadio;
|
||||||
|
import se.sics.cooja.mspmote.interfaces.SkyCoffeeFilesystem;
|
||||||
import se.sics.cooja.mspmote.interfaces.SkyFlash;
|
import se.sics.cooja.mspmote.interfaces.SkyFlash;
|
||||||
import se.sics.cooja.mspmote.interfaces.SkyLED;
|
import se.sics.cooja.mspmote.interfaces.SkyLED;
|
||||||
import se.sics.cooja.mspmote.interfaces.SkySerial;
|
import se.sics.cooja.mspmote.interfaces.SkySerial;
|
||||||
|
@ -198,6 +199,7 @@ public class SkyMoteType extends MspMoteType {
|
||||||
MspMoteID.class,
|
MspMoteID.class,
|
||||||
SkyButton.class,
|
SkyButton.class,
|
||||||
SkyFlash.class,
|
SkyFlash.class,
|
||||||
|
SkyCoffeeFilesystem.class,
|
||||||
SkyByteRadio.class,
|
SkyByteRadio.class,
|
||||||
SkySerial.class,
|
SkySerial.class,
|
||||||
SkyLED.class
|
SkyLED.class
|
||||||
|
|
|
@ -26,24 +26,38 @@
|
||||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||||
* SUCH DAMAGE.
|
* SUCH DAMAGE.
|
||||||
*
|
*
|
||||||
* $Id: CoojaM25P80.java,v 1.1 2008/04/01 08:07:04 fros4943 Exp $
|
* $Id: CoojaM25P80.java,v 1.2 2009/08/11 17:09:16 fros4943 Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package se.sics.cooja.mspmote.interfaces;
|
package se.sics.cooja.mspmote.interfaces;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
|
|
||||||
|
import se.sics.coffee.CoffeeConfiguration;
|
||||||
|
import se.sics.coffee.CoffeeImage;
|
||||||
import se.sics.mspsim.chip.M25P80;
|
import se.sics.mspsim.chip.M25P80;
|
||||||
import se.sics.mspsim.core.*;
|
import se.sics.mspsim.core.*;
|
||||||
|
|
||||||
public class CoojaM25P80 extends M25P80 {
|
public class CoojaM25P80 extends M25P80 implements CoffeeImage {
|
||||||
private static Logger logger = Logger.getLogger(CoojaM25P80.class);
|
private static Logger logger = Logger.getLogger(CoojaM25P80.class);
|
||||||
|
|
||||||
public static int SIZE = 1024*1024;
|
public static int SIZE = 1024*1024;
|
||||||
private byte[] data = new byte[SIZE];
|
private byte[] data = new byte[SIZE];
|
||||||
private long pos;
|
private long pos;
|
||||||
|
|
||||||
|
private static CoffeeConfiguration COFFEE_CONF;
|
||||||
|
static {
|
||||||
|
/* XXX Current implementation only allows for a single coffee configuration */
|
||||||
|
try {
|
||||||
|
COFFEE_CONF = new CoffeeConfiguration("sky.properties");
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public CoojaM25P80(MSP430Core cpu) {
|
public CoojaM25P80(MSP430Core cpu) {
|
||||||
super(cpu);
|
super(cpu);
|
||||||
pos = 0;
|
pos = 0;
|
||||||
|
@ -62,4 +76,36 @@ public class CoojaM25P80 extends M25P80 {
|
||||||
System.arraycopy(b, 0, data, (int) pos, b.length);
|
System.arraycopy(b, 0, data, (int) pos, b.length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* XXX Coffee specific: uses start offset
|
||||||
|
* @see se.sics.coffee.CoffeeImage#erase(int, int)
|
||||||
|
*/
|
||||||
|
public void erase(int size, int offset) throws IOException {
|
||||||
|
Arrays.fill(data, CoffeeConfiguration.startOffset + offset, size, (byte)0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* XXX Coffee specific: uses start offset
|
||||||
|
* @see se.sics.coffee.CoffeeImage#getConfiguration()
|
||||||
|
*/
|
||||||
|
public CoffeeConfiguration getConfiguration() {
|
||||||
|
return COFFEE_CONF;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* XXX Coffee specific: uses start offset
|
||||||
|
* @see se.sics.coffee.CoffeeImage#read(byte[], int, int)
|
||||||
|
*/
|
||||||
|
public void read(byte[] bytes, int size, int offset) throws IOException {
|
||||||
|
System.arraycopy(data, CoffeeConfiguration.startOffset + offset, bytes, 0, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* XXX Coffee specific: uses start offset
|
||||||
|
* @see se.sics.coffee.CoffeeImage#write(byte[], int, int)
|
||||||
|
*/
|
||||||
|
public void write(byte[] bytes, int size, int offset) throws IOException {
|
||||||
|
System.arraycopy(bytes, 0, data, CoffeeConfiguration.startOffset + offset, size);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,315 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2009, Swedish Institute of Computer Science.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
* 1. Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* 3. Neither the name of the Institute nor the names of its contributors
|
||||||
|
* may be used to endorse or promote products derived from this software
|
||||||
|
* without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
|
||||||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||||
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
||||||
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||||
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||||
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||||||
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||||
|
* SUCH DAMAGE.
|
||||||
|
*
|
||||||
|
* $Id: SkyCoffeeFilesystem.java,v 1.1 2009/08/11 17:09:16 fros4943 Exp $
|
||||||
|
*/
|
||||||
|
|
||||||
|
package se.sics.cooja.mspmote.interfaces;
|
||||||
|
|
||||||
|
import java.awt.BorderLayout;
|
||||||
|
import java.awt.EventQueue;
|
||||||
|
import java.awt.Font;
|
||||||
|
import java.awt.event.ActionEvent;
|
||||||
|
import java.awt.event.ActionListener;
|
||||||
|
import java.awt.event.MouseEvent;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Collection;
|
||||||
|
|
||||||
|
import javax.swing.Box;
|
||||||
|
import javax.swing.JButton;
|
||||||
|
import javax.swing.JFileChooser;
|
||||||
|
import javax.swing.JLabel;
|
||||||
|
import javax.swing.JOptionPane;
|
||||||
|
import javax.swing.JPanel;
|
||||||
|
import javax.swing.JTable;
|
||||||
|
import javax.swing.SwingUtilities;
|
||||||
|
import javax.swing.table.AbstractTableModel;
|
||||||
|
|
||||||
|
import org.apache.log4j.Logger;
|
||||||
|
import org.jdom.Element;
|
||||||
|
|
||||||
|
import se.sics.coffee.CoffeeFS;
|
||||||
|
import se.sics.coffee.CoffeeFile;
|
||||||
|
import se.sics.cooja.ClassDescription;
|
||||||
|
import se.sics.cooja.GUI;
|
||||||
|
import se.sics.cooja.Mote;
|
||||||
|
import se.sics.cooja.MoteInterface;
|
||||||
|
import se.sics.cooja.dialogs.TableColumnAdjuster;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Mote user interface to Coffee manager.
|
||||||
|
* Requires users to manually update the filesystem before filesystem operations.
|
||||||
|
*
|
||||||
|
* @author Fredrik Osterlind, Nicolas Tsiftes
|
||||||
|
*/
|
||||||
|
@ClassDescription("Coffee Filesystem")
|
||||||
|
public class SkyCoffeeFilesystem extends MoteInterface {
|
||||||
|
private static Logger logger = Logger.getLogger(SkyCoffeeFilesystem.class);
|
||||||
|
|
||||||
|
private Mote mote;
|
||||||
|
|
||||||
|
private CoffeeFS coffeeFS = null;
|
||||||
|
private CoffeeFile[] files = new CoffeeFile[0];
|
||||||
|
|
||||||
|
private static final int COLUMN_NAME = 0;
|
||||||
|
private static final int COLUMN_SIZE = 1;
|
||||||
|
private static final int COLUMN_SAVE = 2;
|
||||||
|
private static final int COLUMN_REMOVE = 3;
|
||||||
|
private static final String[] COLUMN_NAMES = {
|
||||||
|
"Filename",
|
||||||
|
"Size",
|
||||||
|
"Save",
|
||||||
|
"Remove"
|
||||||
|
};
|
||||||
|
private JTable filesTable;
|
||||||
|
|
||||||
|
public SkyCoffeeFilesystem(Mote mote) {
|
||||||
|
this.mote = mote;
|
||||||
|
|
||||||
|
/* Coffee configuration is already loaded statically */
|
||||||
|
|
||||||
|
filesTable = new JTable(tableModel);
|
||||||
|
filesTable.setFillsViewportHeight(true);
|
||||||
|
filesTable.setFont(new Font("Monospaced", Font.PLAIN, 12));
|
||||||
|
TableColumnAdjuster adjuster = new TableColumnAdjuster(filesTable);
|
||||||
|
adjuster.setDynamicAdjustment(true);
|
||||||
|
adjuster.packColumns();
|
||||||
|
}
|
||||||
|
|
||||||
|
public double energyConsumption() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Collection<Element> getConfigXML() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
public void setConfigXML(Collection<Element> configXML, boolean visAvailable) {
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateFS() {
|
||||||
|
if (SwingUtilities.isEventDispatchThread()) {
|
||||||
|
/* Don't call me in EDT */
|
||||||
|
new Thread(new Runnable() {
|
||||||
|
public void run() {
|
||||||
|
updateFS();
|
||||||
|
}
|
||||||
|
}).start();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Create new filesystem instance */
|
||||||
|
try {
|
||||||
|
SkyFlash flash = mote.getInterfaces().getInterfaceOfType(SkyFlash.class);
|
||||||
|
coffeeFS = new CoffeeFS(flash.m24p80);
|
||||||
|
} catch (IOException e) {
|
||||||
|
coffeeFS = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final CoffeeFile[] tmpFiles = coffeeFS.getFiles().values().toArray(new CoffeeFile[0]);
|
||||||
|
for (CoffeeFile file : tmpFiles) {
|
||||||
|
file.getName();
|
||||||
|
try {
|
||||||
|
file.getLength();
|
||||||
|
} catch (IOException e) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
EventQueue.invokeLater(new Runnable() {
|
||||||
|
public void run() {
|
||||||
|
/* Update table */
|
||||||
|
files = tmpFiles;
|
||||||
|
((AbstractTableModel)filesTable.getModel()).fireTableDataChanged();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public JPanel getInterfaceVisualizer() {
|
||||||
|
JPanel main = new JPanel(new BorderLayout());
|
||||||
|
|
||||||
|
updateFS(); /* Update filesystem */
|
||||||
|
if (coffeeFS == null) {
|
||||||
|
main = new JPanel();
|
||||||
|
main.add(new JLabel("Error when parsing Coffee filesystem"));
|
||||||
|
return main;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Files listing */
|
||||||
|
JPanel panel = new JPanel(new BorderLayout());
|
||||||
|
panel.add(filesTable.getTableHeader(), BorderLayout.NORTH);
|
||||||
|
panel.add(filesTable, BorderLayout.CENTER);
|
||||||
|
main.add(panel, BorderLayout.CENTER);
|
||||||
|
filesTable.repaint();
|
||||||
|
|
||||||
|
|
||||||
|
/* Update (force) */
|
||||||
|
Box box = Box.createHorizontalBox();
|
||||||
|
JButton update = new JButton("Update filesystem");
|
||||||
|
update.addActionListener(new ActionListener() {
|
||||||
|
public void actionPerformed(ActionEvent e) {
|
||||||
|
logger.info("Updating Coffee filesystem");
|
||||||
|
updateFS();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
box.add(update);
|
||||||
|
|
||||||
|
/* Insert */
|
||||||
|
JButton insert = new JButton("Insert file");
|
||||||
|
insert.addActionListener(new ActionListener() {
|
||||||
|
public void actionPerformed(ActionEvent e) {
|
||||||
|
|
||||||
|
JFileChooser fileChooser = new JFileChooser();
|
||||||
|
int reply = fileChooser.showOpenDialog(GUI.getTopParentContainer());
|
||||||
|
if (reply != JFileChooser.APPROVE_OPTION) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final File file = fileChooser.getSelectedFile();
|
||||||
|
new Thread(new Runnable() {
|
||||||
|
public void run() {
|
||||||
|
logger.info("Adding file: " + file.getName());
|
||||||
|
try {
|
||||||
|
coffeeFS.insertFile(file);
|
||||||
|
} catch (IOException e1) {
|
||||||
|
logger.fatal("Coffee exception: " + e1.getMessage(), e1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
updateFS();
|
||||||
|
}
|
||||||
|
}).start();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
box.add(insert);
|
||||||
|
|
||||||
|
main.add(box, BorderLayout.SOUTH);
|
||||||
|
|
||||||
|
return main;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void releaseInterfaceVisualizer(JPanel panel) {
|
||||||
|
}
|
||||||
|
|
||||||
|
private AbstractTableModel tableModel = new AbstractTableModel() {
|
||||||
|
public String getColumnName(int col) {
|
||||||
|
return COLUMN_NAMES[col].toString();
|
||||||
|
}
|
||||||
|
public int getRowCount() {
|
||||||
|
return files.length;
|
||||||
|
}
|
||||||
|
public int getColumnCount() {
|
||||||
|
return COLUMN_NAMES.length;
|
||||||
|
}
|
||||||
|
public Object getValueAt(int row, int col) {
|
||||||
|
if (col == COLUMN_NAME) {
|
||||||
|
return files[row].getName();
|
||||||
|
}
|
||||||
|
if (col == COLUMN_SIZE) {
|
||||||
|
try {
|
||||||
|
return files[row].getLength() + " bytes";
|
||||||
|
} catch (IOException e) {
|
||||||
|
return "? bytes";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return new Boolean(false);
|
||||||
|
}
|
||||||
|
public boolean isCellEditable(int row, int col){
|
||||||
|
return getColumnClass(col) == Boolean.class;
|
||||||
|
}
|
||||||
|
public void setValueAt(Object value, final int row, int col) {
|
||||||
|
if (col == COLUMN_SAVE) {
|
||||||
|
JFileChooser fc = new JFileChooser();
|
||||||
|
int returnVal = fc.showSaveDialog(GUI.getTopParentContainer());
|
||||||
|
if (returnVal != JFileChooser.APPROVE_OPTION) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final File saveFile = fc.getSelectedFile();
|
||||||
|
if (saveFile.exists()) {
|
||||||
|
String s1 = "Overwrite";
|
||||||
|
String s2 = "Cancel";
|
||||||
|
Object[] options = { s1, s2 };
|
||||||
|
int n = JOptionPane.showOptionDialog(
|
||||||
|
GUI.getTopParentContainer(),
|
||||||
|
"A file with the same name already exists.\nDo you want to remove it?",
|
||||||
|
"Overwrite existing file?", JOptionPane.YES_NO_OPTION,
|
||||||
|
JOptionPane.QUESTION_MESSAGE, null, options, s1);
|
||||||
|
if (n != JOptionPane.YES_OPTION) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (saveFile.exists() && !saveFile.canWrite()) {
|
||||||
|
logger.fatal("No write access to file: " + saveFile);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
new Thread(new Runnable() {
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
logger.info("Saving to file: " + saveFile.getName());
|
||||||
|
boolean ok = coffeeFS.extractFile(files[row].getName(), saveFile);
|
||||||
|
if (!ok) {
|
||||||
|
logger.warn("Error when saving to file: " + saveFile.getName());
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.fatal("Coffee exception: " + e.getMessage(), e);
|
||||||
|
}
|
||||||
|
updateFS();
|
||||||
|
}
|
||||||
|
}).start();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (col == COLUMN_REMOVE) {
|
||||||
|
int reply = JOptionPane.showConfirmDialog(
|
||||||
|
GUI.getTopParentContainer(),
|
||||||
|
"Remove \"" + files[row].getName() + "\" from filesystem?");
|
||||||
|
if (reply != JOptionPane.YES_OPTION) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove file */
|
||||||
|
new Thread(new Runnable() {
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
logger.info("Removing file: " + files[row].getName());
|
||||||
|
coffeeFS.removeFile(files[row].getName());
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.fatal("Coffee exception: " + e.getMessage(), e);
|
||||||
|
}
|
||||||
|
updateFS();
|
||||||
|
}
|
||||||
|
}).start();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public Class<?> getColumnClass(int c) {
|
||||||
|
return getValueAt(0, c).getClass();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
|
@ -26,7 +26,7 @@
|
||||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||||
* SUCH DAMAGE.
|
* SUCH DAMAGE.
|
||||||
*
|
*
|
||||||
* $Id: SkyFlash.java,v 1.5 2009/06/17 13:05:27 fros4943 Exp $
|
* $Id: SkyFlash.java,v 1.6 2009/08/11 17:09:16 fros4943 Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package se.sics.cooja.mspmote.interfaces;
|
package se.sics.cooja.mspmote.interfaces;
|
||||||
|
@ -53,7 +53,7 @@ public class SkyFlash extends MoteInterface {
|
||||||
public int SIZE = 1024*1024;
|
public int SIZE = 1024*1024;
|
||||||
|
|
||||||
private SkyMote mote = null;
|
private SkyMote mote = null;
|
||||||
private CoojaM25P80 m24p80 = null;
|
protected CoojaM25P80 m24p80 = null;
|
||||||
|
|
||||||
public SkyFlash(Mote mote) {
|
public SkyFlash(Mote mote) {
|
||||||
this.mote = (SkyMote) mote;
|
this.mote = (SkyMote) mote;
|
||||||
|
|
Loading…
Reference in a new issue