Skip to content

Commit 391bbbc

Browse files
committedMay 13, 2024
8330584: IGV: XML does not save all node properties
Reviewed-by: rcastanedalo, chagedorn
1 parent adaa509 commit 391bbbc

File tree

3 files changed

+131
-68
lines changed

3 files changed

+131
-68
lines changed
 

‎src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/OutlineTopComponent.java

+106-58
Original file line numberDiff line numberDiff line change
@@ -41,16 +41,17 @@
4141
import java.awt.Dimension;
4242
import java.io.*;
4343
import java.nio.channels.FileChannel;
44-
import java.nio.file.Files;
45-
import java.nio.file.Path;
46-
import java.nio.file.Paths;
47-
import java.nio.file.StandardOpenOption;
44+
import java.nio.file.*;
4845
import java.util.*;
4946
import java.util.concurrent.atomic.AtomicBoolean;
5047
import java.util.concurrent.atomic.AtomicInteger;
48+
import java.util.zip.ZipEntry;
49+
import java.util.zip.ZipOutputStream;
50+
import java.util.zip.ZipInputStream;
5151
import javax.swing.*;
5252
import javax.swing.border.Border;
5353
import javax.swing.filechooser.FileFilter;
54+
import javax.swing.filechooser.FileNameExtensionFilter;
5455
import org.netbeans.api.progress.ProgressHandle;
5556
import org.netbeans.api.progress.ProgressHandleFactory;
5657
import org.openide.ErrorManager;
@@ -66,6 +67,9 @@
6667
import org.openide.windows.Mode;
6768
import org.openide.windows.TopComponent;
6869
import org.openide.windows.WindowManager;
70+
import java.nio.channels.Channels;
71+
import java.nio.channels.ReadableByteChannel;
72+
6973

7074
/**
7175
*
@@ -76,17 +80,7 @@ public final class OutlineTopComponent extends TopComponent implements ExplorerM
7680
public static final String PREFERRED_ID = "OutlineTopComponent";
7781
private static final GraphDocument document = new GraphDocument();
7882
private static final int WORK_UNITS = 10000;
79-
private static final FileFilter xmlFileFilter = new FileFilter() {
80-
@Override
81-
public boolean accept(File f) {
82-
return f.getName().toLowerCase().endsWith(".xml") || f.isDirectory();
83-
}
84-
85-
@Override
86-
public String getDescription() {
87-
return "Graph files (*.xml)";
88-
}
89-
};
83+
private static final FileFilter graphFileFilter = new FileNameExtensionFilter("Graph files (*.xml, *.igv)", "xml", "igv");
9084
private static final Server server = new Server(document, OutlineTopComponent::loadContext);
9185
public static OutlineTopComponent instance;
9286
private final Set<FolderNode> selectedFolders = new HashSet<>();
@@ -161,8 +155,26 @@ private static void saveGraphDocument(GraphDocument doc, String path) throws IOE
161155
}
162156
}
163157

164-
try (Writer writer = new OutputStreamWriter(new FileOutputStream(path))) {
165-
Printer.exportGraphDocument(writer, doc, saveContexts);
158+
if (path.endsWith(".igv")) {
159+
File zipFile = new File(path);
160+
String fileName = zipFile.getName();
161+
try (FileOutputStream fos = new FileOutputStream(zipFile);
162+
ZipOutputStream zos = new ZipOutputStream(fos);
163+
Writer writer = new OutputStreamWriter(zos)) {
164+
165+
// Replace the '.igv' extension with '.xml's
166+
String zipEntryName = fileName.substring(0, fileName.length() - 4) + ".xml";
167+
ZipEntry zipEntry = new ZipEntry(zipEntryName);
168+
zos.putNextEntry(zipEntry);
169+
170+
Printer.exportGraphDocument(writer, doc, saveContexts);
171+
172+
zos.closeEntry();
173+
}
174+
} else {
175+
try (Writer writer = new OutputStreamWriter(new FileOutputStream(path))) {
176+
Printer.exportGraphDocument(writer, doc, saveContexts);
177+
}
166178
}
167179
}
168180

@@ -358,7 +370,7 @@ public void clearWorkspace() {
358370
**/
359371
public void openFile() {
360372
JFileChooser fc = new JFileChooser(Settings.get().get(Settings.DIRECTORY, Settings.DIRECTORY_DEFAULT));
361-
fc.setFileFilter(xmlFileFilter);
373+
fc.setFileFilter(graphFileFilter);
362374
if (fc.showOpenDialog(null) == JFileChooser.APPROVE_OPTION) {
363375
clearWorkspace();
364376
String path = fc.getSelectedFile().getAbsolutePath();
@@ -403,9 +415,22 @@ public void save() {
403415
}
404416

405417
public void saveAs() {
406-
JFileChooser fc = new JFileChooser();
418+
JFileChooser fc = new JFileChooser() {
419+
@Override
420+
public void approveSelection() {
421+
File selectedFile = getSelectedFile();
422+
if (selectedFile != null) {
423+
String fileName = selectedFile.getName().toLowerCase();
424+
if (!fileName.endsWith(".xml") && !fileName.endsWith(".igv")) {
425+
JOptionPane.showMessageDialog(this, "Please select a graph file with .xml or .igv extension.", "Invalid File", JOptionPane.ERROR_MESSAGE);
426+
return;
427+
}
428+
}
429+
super.approveSelection();
430+
}
431+
};
407432
fc.setDialogTitle("Save As...");
408-
fc.setFileFilter(xmlFileFilter);
433+
fc.setFileFilter(graphFileFilter);
409434
fc.setCurrentDirectory(new File(Settings.get().get(Settings.DIRECTORY, Settings.DIRECTORY_DEFAULT)));
410435
if (fc.showSaveDialog(null) == JFileChooser.APPROVE_OPTION) {
411436
String path = fc.getSelectedFile().getAbsolutePath();
@@ -432,7 +457,7 @@ public void saveAs() {
432457
**/
433458
public void importFromXML() {
434459
JFileChooser fc = new JFileChooser();
435-
fc.setFileFilter(xmlFileFilter);
460+
fc.setFileFilter(graphFileFilter);
436461
fc.setCurrentDirectory(new File(Settings.get().get(Settings.DIRECTORY, Settings.DIRECTORY_DEFAULT)));
437462
fc.setMultiSelectionEnabled(true);
438463
if (fc.showOpenDialog(null) == JFileChooser.APPROVE_OPTION) {
@@ -482,60 +507,83 @@ private static void loadContext(GraphContext context) {
482507
}
483508

484509
/**
485-
* Loads a graph document from the given file path, updating progress via a ProgressHandle.
486-
* Parse the XML file, add the parsed document to the workspace, and load associated contexts if specified.
510+
* Loads a graph document from the specified path, either as an XML file or from a ZIP archive.
511+
* If loading the context is requested, it loads the context along with the document.
487512
*/
488513
private void loadGraphDocument(String path, boolean loadContext) throws IOException {
489514
if (Files.notExists(Path.of(path))) {
490515
return;
491516
}
492517
File file = new File(path);
493-
final FileChannel channel;
494-
final long start;
495-
try {
496-
channel = FileChannel.open(file.toPath(), StandardOpenOption.READ);
497-
start = channel.size();
498-
} catch (Exception ex) {
499-
Exceptions.printStackTrace(ex);
500-
return;
518+
if (file.getName().endsWith(".xml")) {
519+
try (FileChannel channel = FileChannel.open(file.toPath(), StandardOpenOption.READ)) {
520+
loadFile(channel, file, loadContext);
521+
}
522+
} else if (file.getName().endsWith(".igv")) {
523+
try (ZipInputStream zis = new ZipInputStream(new FileInputStream(file))) {
524+
ZipEntry entry = zis.getNextEntry();
525+
if (entry != null && entry.getName().endsWith(".xml")) {
526+
loadFile(Channels.newChannel(zis), file, loadContext);
527+
}
528+
}
501529
}
530+
}
502531

532+
/**
533+
* Loads an XML or ZIP document from the provided channel, while monitoring the progress of the operation.
534+
*/
535+
private void loadFile(ReadableByteChannel channel, File file, boolean loadContext) throws IOException {
503536
final ProgressHandle handle = ProgressHandleFactory.createHandle("Opening file " + file.getName());
504537
handle.start(WORK_UNITS);
505538

506-
ParseMonitor monitor = new ParseMonitor() {
507-
@Override
508-
public void updateProgress() {
509-
try {
510-
int prog = (int) (WORK_UNITS * (double) channel.position() / (double) start);
511-
handle.progress(prog);
512-
} catch (IOException ignored) {
539+
ParseMonitor monitor;
540+
if (channel instanceof FileChannel fileChannel) {
541+
final long start = fileChannel.size();
542+
monitor = new ParseMonitor() {
543+
@Override
544+
public void updateProgress() {
545+
try {
546+
int prog = (int) (WORK_UNITS * (double) fileChannel.position() / (double) start);
547+
handle.progress(prog);
548+
} catch (IOException ignored) {}
513549
}
514-
}
515550

516-
@Override
517-
public void setState(String state) {
518-
updateProgress();
519-
handle.progress(state);
520-
}
521-
};
551+
@Override
552+
public void setState(String state) {
553+
updateProgress();
554+
handle.progress(state);
555+
}
556+
};
557+
} else {
558+
monitor = new ParseMonitor() {
559+
@Override
560+
public void updateProgress() {
561+
handle.progress("Processing...");
562+
}
563+
564+
@Override
565+
public void setState(String state) {
566+
updateProgress();
567+
handle.progress(state);
568+
}
569+
};
570+
}
571+
522572
try {
523-
if (file.getName().endsWith(".xml")) {
524-
ArrayList<GraphContext> contexts = new ArrayList<>();
525-
final Parser parser = new Parser(channel, monitor, document, loadContext ? contexts::add : null);
526-
parser.parse();
527-
SwingUtilities.invokeLater(() -> {
528-
for (Node child : manager.getRootContext().getChildren().getNodes(true)) {
529-
// Nodes are lazily created. By expanding and collapsing they are all initialized
530-
((BeanTreeView) this.treeView).expandNode(child);
531-
((BeanTreeView) this.treeView).collapseNode(child);
532-
}
533-
requestActive();
534-
});
573+
ArrayList<GraphContext> contexts = new ArrayList<>();
574+
final Parser parser = new Parser(channel, monitor, document, loadContext ? contexts::add : null);
575+
parser.parse();
576+
SwingUtilities.invokeLater(() -> {
577+
for (Node child : manager.getRootContext().getChildren().getNodes(true)) {
578+
// Nodes are lazily created. By expanding and collapsing they are all initialized
579+
((BeanTreeView) this.treeView).expandNode(child);
580+
((BeanTreeView) this.treeView).collapseNode(child);
581+
}
582+
requestActive();
535583
for (GraphContext ctx : contexts) {
536584
loadContext(ctx);
537585
}
538-
}
586+
});
539587
} catch (IOException ex) {
540588
Exceptions.printStackTrace(ex);
541589
}

‎src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/InputNode.java

+12-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2008, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2008, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -23,6 +23,8 @@
2323
*/
2424
package com.sun.hotspot.igv.data;
2525

26+
import java.util.Objects;
27+
2628
/**
2729
*
2830
* @author Thomas Wuerthinger
@@ -49,17 +51,21 @@ public int getId() {
4951
}
5052

5153
@Override
52-
public boolean equals(Object o) {
53-
if (!(o instanceof InputNode)) {
54+
public boolean equals(Object obj) {
55+
if (this == obj) {
56+
return true;
57+
}
58+
if (obj == null || getClass() != obj.getClass()) {
5459
return false;
5560
}
56-
InputNode n = (InputNode) o;
57-
return n.id == id;
61+
InputNode other = (InputNode) obj;
62+
return id == other.id &&
63+
Objects.equals(getProperties(), other.getProperties());
5864
}
5965

6066
@Override
6167
public int hashCode() {
62-
return id * 13;
68+
return Objects.hash(id, getProperties());
6369
}
6470

6571
@Override

‎src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/serialization/Printer.java

+13-4
Original file line numberDiff line numberDiff line change
@@ -103,10 +103,15 @@ private static void exportInputGraph(XMLWriter writer, InputGraph graph, InputGr
103103
for (InputNode n : removed) {
104104
writer.simpleTag(Parser.REMOVE_NODE_ELEMENT, new Properties(Parser.NODE_ID_PROPERTY, Integer.toString(n.getId())));
105105
}
106-
}
107-
108-
for (InputNode n : graph.getNodes()) {
109-
if (!difference || !equal.contains(n)) {
106+
for (InputNode n : graph.getNodes()) {
107+
if (!equal.contains(n)) {
108+
writer.startTag(Parser.NODE_ELEMENT, new Properties(Parser.NODE_ID_PROPERTY, Integer.toString(n.getId())));
109+
writer.writeProperties(n.getProperties());
110+
writer.endTag(); // Parser.NODE_ELEMENT
111+
}
112+
}
113+
} else {
114+
for (InputNode n : graph.getNodes()) {
110115
writer.startTag(Parser.NODE_ELEMENT, new Properties(Parser.NODE_ID_PROPERTY, Integer.toString(n.getId())));
111116
writer.writeProperties(n.getProperties());
112117
writer.endTag(); // Parser.NODE_ELEMENT
@@ -227,6 +232,10 @@ private static void exportInputMethod(XMLWriter w, InputMethod method) throws IO
227232
b.append(code.getBci());
228233
b.append(" ");
229234
b.append(code.getName());
235+
b.append(" ");
236+
b.append(code.getOperands());
237+
b.append(" ");
238+
b.append(code.getComment());
230239
b.append("\n");
231240
}
232241

0 commit comments

Comments
 (0)
Please sign in to comment.