Utils.java revision 77f2b82a2e80af8da52c22d69a76def6d4209757
1package com.android.server.wifi.hotspot2;
2
3import java.nio.ByteBuffer;
4import java.util.ArrayList;
5import java.util.Calendar;
6import java.util.Collection;
7import java.util.LinkedList;
8import java.util.List;
9import java.util.TimeZone;
10
11import static com.android.server.wifi.anqp.Constants.BYTE_MASK;
12import static com.android.server.wifi.anqp.Constants.NIBBLE_MASK;
13
14public abstract class Utils {
15
16    private static final String[] PLMNText = { "wlan", "mnc*", "mcc*", "3gppnetwork", "org" };
17
18    public static List<String> splitDomain(String domain) {
19
20        if (domain.endsWith("."))
21            domain = domain.substring(0, domain.length() - 1);
22        int at = domain.indexOf('@');
23        if (at >= 0)
24            domain = domain.substring(at + 1);
25
26        String[] labels = domain.toLowerCase().split("\\.");
27        LinkedList<String> labelList = new LinkedList<String>();
28        for (String label : labels) {
29            labelList.addFirst(label);
30        }
31
32        return labelList;
33    }
34
35    public static long parseMac(String s) {
36
37        long mac = 0;
38        int count = 0;
39        for (int n = 0; n < s.length(); n++) {
40            int nibble = Utils.fromHex(s.charAt(n), true);
41            if (nibble >= 0) {
42                mac = (mac << 4) | nibble;
43                count++;
44            }
45        }
46        if (count < 12 || (count&1) == 1) {
47            throw new IllegalArgumentException("Bad MAC address: '" + s + "'");
48        }
49        return mac;
50    }
51
52    public static int[] getMccMnc(List<String> domain) {
53        if (domain.size() != 5) {
54            return null;
55        }
56
57        for (int n = 4; n >= 0; n-- ) {
58            String expect = PLMNText[n];
59            int len = expect.endsWith("*") ? expect.length() - 1 : expect.length();
60            if (!domain.get(n).regionMatches(0, expect, 0, len)) {
61                return null;
62            }
63        }
64        try {
65            int[] mccMnc = new int[2];
66            mccMnc[0] = Integer.parseInt(domain.get(1).substring(3));
67            mccMnc[1] = Integer.parseInt(domain.get(2).substring(3));
68            return mccMnc;
69        }
70        catch (NumberFormatException nfe) {
71            return null;
72        }
73    }
74
75    public static String roamingConsortiumsToString(long[] ois) {
76        if (ois == null) {
77            return "null";
78        }
79        List<Long> list = new ArrayList<Long>(ois.length);
80        for (long oi : ois) {
81            list.add(oi);
82        }
83        return roamingConsortiumsToString(list);
84    }
85
86    public static String roamingConsortiumsToString(Collection<Long> ois) {
87        StringBuilder sb = new StringBuilder();
88        boolean first = true;
89        for (long oi : ois) {
90            if (first) {
91                first = false;
92            } else {
93                sb.append(", ");
94            }
95            if (Long.numberOfLeadingZeros(oi) > 40) {
96                sb.append(String.format("%06x", oi));
97            } else {
98                sb.append(String.format("%010x", oi));
99            }
100        }
101        return sb.toString();
102    }
103
104    public static String toHexString(byte[] data) {
105        if (data == null) {
106            return "null";
107        }
108        StringBuilder sb = new StringBuilder(data.length * 3);
109
110        boolean first = true;
111        for (byte b : data) {
112            if (first) {
113                first = false;
114            } else {
115                sb.append(' ');
116            }
117            sb.append(String.format("%02x", b & BYTE_MASK));
118        }
119        return sb.toString();
120    }
121
122    public static byte[] hexToBytes(String text) {
123        if ((text.length() & 1) == 1) {
124            throw new NumberFormatException("Odd length hex string: " + text.length());
125        }
126        byte[] data = new byte[text.length() >> 1];
127        int position = 0;
128        for (int n = 0; n < text.length(); n += 2) {
129            data[position] =
130                    (byte) (((fromHex(text.charAt(n), false) & NIBBLE_MASK) << 4) |
131                            (fromHex(text.charAt(n + 1), false) & NIBBLE_MASK));
132            position++;
133        }
134        return data;
135    }
136
137    public static int fromHex(char ch, boolean lenient) throws NumberFormatException {
138        if (ch <= '9' && ch >= '0') {
139            return ch - '0';
140        } else if (ch >= 'a' && ch <= 'f') {
141            return ch + 10 - 'a';
142        } else if (ch <= 'F' && ch >= 'A') {
143            return ch + 10 - 'A';
144        } else if (lenient) {
145            return -1;
146        } else {
147            throw new NumberFormatException("Bad hex-character: " + ch);
148        }
149    }
150
151    private static char toAscii(int b) {
152        return b >= ' ' && b < 0x7f ? (char) b : '.';
153    }
154
155    static boolean isDecimal(String s) {
156        for (int n = 0; n < s.length(); n++) {
157            char ch = s.charAt(n);
158            if (ch < '0' || ch > '9') {
159                return false;
160            }
161        }
162        return true;
163    }
164
165    public static <T extends Comparable> int compare(Comparable<T> c1, T c2) {
166        if (c1 == null) {
167            return c2 == null ? 0 : -1;
168        }
169        else if (c2 == null) {
170            return 1;
171        }
172        else {
173            return c1.compareTo(c2);
174        }
175    }
176
177    public static String bytesToBingoCard(ByteBuffer data, int len) {
178        ByteBuffer dup = data.duplicate();
179        dup.limit(dup.position() + len);
180        return bytesToBingoCard(dup);
181    }
182
183    public static String bytesToBingoCard(ByteBuffer data) {
184        ByteBuffer dup = data.duplicate();
185        StringBuilder sbx = new StringBuilder();
186        while (dup.hasRemaining()) {
187            sbx.append(String.format("%02x ", dup.get() & BYTE_MASK));
188        }
189        dup = data.duplicate();
190        sbx.append(' ');
191        while (dup.hasRemaining()) {
192            sbx.append(String.format("%c", toAscii(dup.get() & BYTE_MASK)));
193        }
194        return sbx.toString();
195    }
196
197    public static String toUTCString(long ms) {
198        if (ms < 0) {
199            return "unset";
200        }
201        Calendar c = Calendar.getInstance(TimeZone.getTimeZone("UTC"));
202        c.setTimeInMillis(ms);
203        return String.format("%4d/%02d/%02d %2d:%02d:%02dZ",
204                c.get(Calendar.YEAR),
205                c.get(Calendar.MONTH) + 1,
206                c.get(Calendar.DAY_OF_MONTH),
207                c.get(Calendar.HOUR_OF_DAY),
208                c.get(Calendar.MINUTE),
209                c.get(Calendar.SECOND));
210    }
211}
212