/*
 * Copyright (C) 2011 The Android Open Source Project
 *
 * Licensed 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.
 */

import dalvik.system.PathClassLoader;
import dalvik.system.VMDebug;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.File;
import java.io.IOException;

public class Main {
  public static void main(String[] args) throws Exception {
    System.loadLibrary(args[0]);
    System.out.println("Hello, world!");
    String agent = null;
    // By default allow debugging
    boolean debugging_allowed = true;
    for(String a : args) {
      if(a.startsWith("agent:")) {
        agent = a.substring(6);
      } else if (a.equals("disallow-debugging")) {
        debugging_allowed = false;
      }
    }
    if (agent == null) {
      throw new Error("Could not find agent: argument!");
    }
    setDebuggingAllowed(debugging_allowed);
    // Setup is finished. Try to attach agent in 2 ways.
    try {
      VMDebug.attachAgent(agent);
    } catch(SecurityException e) {
      System.out.println(e.getMessage());
    }
    attachWithClassLoader(args);
    System.out.println("Goodbye!");
  }

  private static native void setDebuggingAllowed(boolean val);

  private static void attachWithClassLoader(String[] args) throws Exception {
    for(String a : args) {
      if(a.startsWith("agent:")) {
        String agentName = a.substring(6, a.indexOf('='));
        File tmp = null;
        try {
          tmp = File.createTempFile("lib", ".so");
          prepare(agentName, tmp);

          String newAgentName = tmp.getName();
          String agent = a.substring(6).replace(agentName, newAgentName);

          ClassLoader cl = new PathClassLoader("", tmp.getParentFile().getAbsolutePath(),
              Main.class.getClassLoader());
          try {
            VMDebug.attachAgent(agent, cl);
          } catch(SecurityException e) {
            System.out.println(e.getMessage());
          }
        } catch (Exception e) {
          e.printStackTrace(System.out);
        } finally {
          if (tmp != null) {
            tmp.delete();
          }
        }
      }
    }
  }

  private static void prepare(String in, File tmp) throws Exception {
    // Find the original.
    File orig = find(in);
    if (orig == null) {
      throw new RuntimeException("Could not find " + in);
    }
    // Copy the original.
    {
      BufferedInputStream bis = new BufferedInputStream(new FileInputStream(orig));
      BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(tmp));
      byte[] buf = new byte[16 * 1024];
      for (;;) {
        int r = bis.read(buf, 0, buf.length);
        if (r < 0) {
          break;
        } else if (r > 0) {
          bos.write(buf, 0, r);
        }
      }
      bos.close();
      bis.close();
    }
  }

  private static File find(String in) {
    String libraryPath = System.getProperty("java.library.path");
    for (String path : libraryPath.split(":")) {
      File f = new File(path + "/" + in);
      if (f.exists()) {
        return f;
      }
    }
    return null;
  }
}