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

org.netbeans.modules.cpplite.debugger.CPPFrame Maven / Gradle / Ivy

/*
 * 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.netbeans.modules.cpplite.debugger;

import java.io.File;
import java.net.MalformedURLException;
import java.net.URI;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.CompletableFuture;
import java.util.logging.Level;
import java.util.logging.Logger;

import org.netbeans.api.annotations.common.CheckForNull;
import org.netbeans.modules.cnd.debugger.gdb2.mi.MIConst;
import org.netbeans.modules.cnd.debugger.gdb2.mi.MIRecord;
import org.netbeans.modules.cnd.debugger.gdb2.mi.MIResult;
import org.netbeans.modules.cnd.debugger.gdb2.mi.MITList;
import org.netbeans.modules.cnd.debugger.gdb2.mi.MITListItem;
import org.netbeans.modules.cnd.debugger.gdb2.mi.MIValue;
import org.netbeans.modules.nativeimage.api.debug.EvaluateException;
import org.netbeans.modules.nativeimage.api.debug.NIFrame;
import org.netbeans.modules.nativeimage.api.debug.NIVariable;
import org.netbeans.modules.nativeimage.spi.debug.filters.FrameDisplayer;
import org.netbeans.modules.nativeimage.spi.debug.filters.FrameDisplayer.DisplayedFrame;
import org.netbeans.spi.debugger.ui.DebuggingView.DVFrame;

import org.openide.cookies.LineCookie;
import org.openide.filesystems.FileObject;
import org.openide.filesystems.FileUtil;
import org.openide.filesystems.URLMapper;
import org.openide.text.Line;
import org.openide.util.Pair;

public final class CPPFrame implements DVFrame {

    private static final Logger LOGGER = Logger.getLogger(CPPFrame.class.getName());

    private final CPPThread thread;
    private final DisplayedFrame displayedFrame;
    private final NIFrame niFrame;
    public final int level;

    private volatile Map variables;

    private CPPFrame(CPPThread thread, DisplayedFrame displayedFrame, NIFrame niFrame) {
        Objects.requireNonNull(thread);
        Objects.requireNonNull(displayedFrame);
        Objects.requireNonNull(niFrame);
        this.thread = thread;
        this.displayedFrame = displayedFrame;
        this.niFrame = niFrame;
        this.level = niFrame.getLevel();
    }

    static CPPFrame create(CPPThread thread, MITList frame) {
        NIFrame niFrame = new NIFrameImpl(thread.getId(), frame);
        FrameDisplayer frameDisplayer = thread.getDebugger().getContextProvider().lookupFirst(null, FrameDisplayer.class);
        DisplayedFrame displayedFrame = frameDisplayer != null ? frameDisplayer.displayed(niFrame) : createDisplayedFrame(niFrame);
        if (displayedFrame == null) {
            return null; // Not to be displayed
        }
        return new CPPFrame(thread, displayedFrame, niFrame);
    }

    NIFrame getFrame() {
        return niFrame;
    }

    @Override
    public String getName() {
        return displayedFrame.getDisplayName();
    }

    public String getDescription() {
        return displayedFrame.getDescription();
    }

    @CheckForNull
    public Line location() {
        URI sourceURI = displayedFrame.getSourceURI();
        if (sourceURI == null) {
            return null;
        }
        FileObject file;
        try {
            file = URLMapper.findFileObject(sourceURI.toURL());
        } catch (MalformedURLException ex) {
            return null;
        }
        if (file == null) {
            return null;
        }
        LineCookie lc = file.getLookup().lookup(LineCookie.class);
        return lc.getLineSet().getOriginal(displayedFrame.getLine() - 1);
    }

    @Override
    public CPPThread getThread() {
        return thread;
    }

    @Override
    public void makeCurrent() {
        thread.getDebugger().setCurrentStackFrame(this);
    }

    @Override
    public URI getSourceURI() {
        return displayedFrame.getSourceURI();
    }

    @Override
    public int getLine() {
        return displayedFrame.getLine();
    }

    @Override
    public int getColumn() {
        return -1;
    }

    public Map getVariables() {
        Map vars = variables;
        if (vars == null) {
            synchronized (this) {
                vars = variables;
                if (vars == null) {
                    variables = vars = retrieveVariables(this, null);
                }
            }
        }
        return vars;
    }

    static Map retrieveVariables(CPPFrame frame, CPPVariable parentVar) {
        MIRecord record;
        try {
            if (parentVar == null) {
                record = frame.thread.getDebugger().sendAndGet("-stack-list-variables --thread " + frame.thread.getId() + " --frame " + frame.level + " --no-frame-filters 2");
            } else {
                // from to
                record = frame.thread.getDebugger().sendAndGet("-var-list-children --thread " + frame.thread.getId() + " --frame " + frame.level + " --all-values " + "\"" + parentVar.getUniqueName() + "\"");
            }
        } catch (InterruptedException ex) {
            return Collections.emptyMap();
        }
        if (record.isError()) {
            return Collections.singletonMap(record.error(), null);
        }
        MITList results = record.results();
        if (results.isEmpty()) {
            return Collections.emptyMap();
        }
        Map map = new LinkedHashMap<>(results.size());
        MIValue children = results.valueOf("children");
        if (children != null) {
            for (MITListItem item : children.asList()) {
                MITList child = ((MIResult) item).value().asTList();
                String uniqueName = child.getConstValue("name");
                String name = child.getConstValue("exp");
                int numChildren = Integer.parseInt(child.getConstValue("numchild"));
                String type = child.getConstValue("type");
                MIValue value = child.valueOf("value");
                map.put(name, new CPPVariable(frame, parentVar, uniqueName, name, type, value, numChildren));
            }
        } else {
            MIValue resultValue = results.valueOf("variables");
            if (resultValue == null) {
                return Collections.emptyMap();
            } else if (resultValue.isConst()) {
                return Collections.singletonMap(((MIConst) resultValue).value(), null);
            }
            results = (MITList) resultValue;
            LOGGER.log(Level.FINE, "retrieveVariables: have {0} variables:", results.size());
            for (MITListItem item : results) {
                MITList varList = (MITList) item;
                String name = varList.getConstValue("name");
                String type = varList.getConstValue("type");
                MIValue value = varList.valueOf("value");
                Pair uniqueVar = createVariable(frame, name);
                String uniqueName = uniqueVar.first();
                if (uniqueName != null) {
                    int numChildren = uniqueVar.second();
                    LOGGER.log(Level.FINE, "  {0} = ({1}) {2} ; [{3}]", new Object[]{name, type, value, numChildren});
                    map.put(name, new CPPVariable(frame, parentVar, uniqueName, name, type, value, numChildren));
                }
            }
        }
        return map;
    }

    private static Pair createVariable(CPPFrame frame, String variableName) {
        String uniqueName = null;
        int numChildren = 0;
        MIRecord record;
        try {
            record = frame.thread.getDebugger().sendAndGet("-var-create --thread " + frame.thread.getId() + " --frame " + frame.level + " - " + "*" + " " + variableName);
            if (!record.isError() && !record.isEmpty()) {
                String name = record.results().getConstValue("name");
                if (!name.isEmpty()) {
                    uniqueName = name;
                    String numchild = record.results().getConstValue("numchild");
                    numChildren = Integer.parseInt(numchild);
                }
            }
        } catch (InterruptedException ex) {
        }
        return Pair.of(uniqueName, numChildren);
    }

    private static int retrieveNumChildren(CPPFrame frame, String variableName) {
        int numChildren = 0;
        MIRecord record;
        try {
            record = frame.thread.getDebugger().sendAndGet("-var-info-num-children --thread " + frame.thread.getId() + " --frame " + frame.level + " " + variableName);
            if (!record.isError() && !record.isEmpty()) {
                String numchild = record.results().getConstValue("numchild");
                if (!numchild.isEmpty()) {
                    numChildren = Integer.parseInt(numchild);
                }
            }
        } catch (InterruptedException ex) {
        }
        return numChildren;
    }

    private static final String MI_ERROR = "MI parse error: ";

    public CompletableFuture evaluateAsync(String expression) {
        return evaluateAsync(expression, null);
    }

    public CompletableFuture evaluateAsync(String expression, String resultName) {
        String resultVarName = (resultName != null) ? resultName : expression;
        CompletableFuture result = new CompletableFuture<>();
        NIVariable value = getVariables().get(expression);
        if (value != null) {
            result.complete(value);
            return result;
        }
        thread.getDebugger().send(new Command("-var-create --thread " + thread.getId() + " --frame " + level + " - * \"" + expression + "\"") {
            @Override
            protected void onDone(MIRecord record) {
                MITList results = record.results();
                String varName = results.valueOf("name").asConst().value();
                MIValue typeValue = results.valueOf("type");
                String type = typeValue != null ? typeValue.asConst().value() : null;
                MIValue resultValue = results.valueOf("value");
                int numChildren;
                MIValue numchildValue = results.valueOf("numchild");
                if (numchildValue != null) {
                    numChildren = Integer.parseInt(numchildValue.asConst().value());
                } else {
                    numChildren = retrieveNumChildren(CPPFrame.this, varName);
                }
                result.complete(new CPPVariable(CPPFrame.this, null, varName, resultVarName, type, resultValue, numChildren));
                //thread.getDebugger().send(new Command("-var-delete " + varName));
            }
            @Override
            protected void onError(MIRecord record) {
                String error = record.error();
                if (error.startsWith(MI_ERROR)) {
                    error = error.substring(MI_ERROR.length());
                    result.completeExceptionally(new EvaluateException(error));
                }
            }
        });
        return result;
    }

    private static DisplayedFrame createDisplayedFrame(NIFrame frame) {
        return DisplayedFrame.newBuilder(getDisplayName(frame))
                .description(getDescription(frame))
                .line(frame.getLine())
                .sourceURISupplier(() -> getSourceURI(frame))
                .build();
    }

    private static String getDisplayName(NIFrame frame) {
        StringBuilder builder = new StringBuilder(frame.getFunctionName());
        String shortName = frame.getShortFileName();
        if (shortName != null) {
            builder.append("; ");
            builder.append(shortName);
        }
        int line = frame.getLine();
        if (line > 0) {
            builder.append(':');
            builder.append(line);
        }
        return builder.toString();
    }

    private static String getDescription(NIFrame frame) {
        StringBuilder builder = new StringBuilder(frame.getFunctionName());
        String fullName = frame.getFullFileName();
        if (fullName != null) {
            builder.append("; ");
            builder.append(fullName);
        }
        int line = frame.getLine();
        if (line > 0) {
            builder.append(':');
            builder.append(line);
        }
        return builder.toString();
    }

    private static URI getSourceURI(NIFrame frame) {
        String fullFileName = frame.getFullFileName();
        if (fullFileName != null && !fullFileName.isEmpty()) {
            FileObject file = FileUtil.toFileObject(FileUtil.normalizeFile(new File(fullFileName)));
            if (file != null) {
                return file.toURI();
            }
        }
        return null;
    }

    private static final class NIFrameImpl implements NIFrame {

        private final String threadId;
        private final int level;
        private final String address;
        private final String shortFileName;
        private final String fullFileName;
        private final String functionName;
        private final int line;

        NIFrameImpl(String threadId, MITList frame) {
            this.threadId = threadId;
            this.address = frame.getConstValue("addr");
            this.shortFileName = frame.valueOf("file") != null ? frame.valueOf("file").asConst().value() : null;
            this.functionName = frame.valueOf("func").asConst().value();
            this.fullFileName = frame.valueOf("fullname") != null ? frame.valueOf("fullname").asConst().value() : null;
            this.line = frame.valueOf("line") != null ? Integer.parseInt(frame.valueOf("line").asConst().value()) : -1;
            if (frame.valueOf("level") != null) {
                this.level = Integer.parseInt(frame.valueOf("level").asConst().value());
            } else {
                this.level = 0;
            }
        }

        @Override
        public String getAddress() {
            return address;
        }

        @Override
        public String getShortFileName() {
            return shortFileName;
        }

        @Override
        public String getFullFileName() {
            return fullFileName;
        }

        @Override
        public String getFunctionName() {
            return functionName;
        }

        @Override
        public int getLine() {
            return line;
        }

        @Override
        public String getThreadId() {
            return threadId;
        }

        @Override
        public int getLevel() {
            return level;
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy