import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.HashMap;
public class Main {
    private static final BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
    public static int readInt() throws IOException {
        return Integer.parseInt(reader.readLine());
    }
    public static int[] readIntArr() throws IOException {
        return Arrays.stream(reader.readLine().split(" ")).mapToInt(Integer::parseInt).toArray();
    }
    public static long readLong() throws IOException {
        return Long.parseLong(reader.readLine());
    }
    public static long[] readLongs() throws IOException {
        return Arrays.stream(reader.readLine().split(" ")).mapToLong(Long::parseLong).toArray();
    }
    public static <T> void print(T str) {
        System.out.print(str);
    }
    public static void main(String[] args) throws IOException {
        int n = 1; // readInt();
        for (int i = 1; i <= n; i++) {
            solve();
        }
    }
    public static void solve() throws IOException {
        long[] Ns = readLongs();
        int nJobs = (int)Ns[0]; long initMoney = Ns[0];
        long[] profitMap = new long[nJobs+1];
        ArrayList<Integer>[] adj = new ArrayList[nJobs+1];
        profitMap[0] = 0;
        adj[0] = new ArrayList<>();
        for (int i = 1; i <= nJobs; i++) {
            adj[i] = new ArrayList<>();
            int[] xp = readIntArr();
            profitMap[i] = xp[0];
            //PRE xp[1] < i
            adj[xp[1]].add(i);
        }
        print(maxProfit(0,initMoney,profitMap,adj));
    }
    public static long maxProfit(int job, long money, long[] profitMap, ArrayList<Integer>[] adj) {
        if (adj[job].isEmpty()) { return Math.max(profitMap[job],0); }
        HashMap<Integer,Long> maxProfitMap = new HashMap<>();
        long sumProfit = 0;
        while (true) {
            var oldMaxProfitMap = (HashMap<Integer,Long>)maxProfitMap.clone();
            for (int preJob : adj[job]) {
                maxProfitMap.put(preJob,maxProfit(preJob, money + profitMap[job], profitMap, adj));
            }
            adj[job].sort(Comparator.comparingLong(o -> -maxProfitMap.get(o)+profitMap[o]));
            var gaining = false;
            for (int i = 0; i < adj[job].size(); i++) {
                int j = adj[job].get(i);
                if (money + profitMap[j] < 0) { continue; }
                long gain = maxProfitMap.get(j)
                        - (oldMaxProfitMap.containsKey(j) ? oldMaxProfitMap.get(j) : 0);
                if (gain <= 0) { continue; }
                gaining = true;
                sumProfit += gain;
                money += gain;
            }
            if (!gaining) { break; }
        }
        return sumProfit + profitMap[job];
    }
}
Compilation message (stderr)
Note: Main.java uses unchecked or unsafe operations.
Note: Recompile with -Xlint:unchecked for details.
=======
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... |