diff options
author | Jeff Sharkey <jsharkey@android.com> | 2011-05-13 19:48:11 -0700 |
---|---|---|
committer | Jeff Sharkey <jsharkey@android.com> | 2011-05-13 19:48:11 -0700 |
commit | 5637884f8217dbf28de0913c06aceaf09c898510 (patch) | |
tree | 48564b01ad7a5456a2ac4d937945b0ba3fbe8bfa /dalvik | |
parent | aaa0003924a16a06eb871456c33364461c25a834 (diff) | |
parent | 1e8d508f2d97e19f2fc8a709330ea97e1e9f203a (diff) | |
download | libcore-5637884f8217dbf28de0913c06aceaf09c898510.zip libcore-5637884f8217dbf28de0913c06aceaf09c898510.tar.gz libcore-5637884f8217dbf28de0913c06aceaf09c898510.tar.bz2 |
resolved conflicts for merge of 1e8d508f to dalvik-dev
Change-Id: Id0739d4e567b112cbea48f5ecb7dd46425da1a36
Diffstat (limited to 'dalvik')
-rw-r--r-- | dalvik/src/main/java/dalvik/system/BlockGuard.java | 95 |
1 files changed, 92 insertions, 3 deletions
diff --git a/dalvik/src/main/java/dalvik/system/BlockGuard.java b/dalvik/src/main/java/dalvik/system/BlockGuard.java index 6744448..64699ff 100644 --- a/dalvik/src/main/java/dalvik/system/BlockGuard.java +++ b/dalvik/src/main/java/dalvik/system/BlockGuard.java @@ -16,17 +16,21 @@ package dalvik.system; +import java.io.ByteArrayOutputStream; import java.io.FileDescriptor; -import java.io.FileNotFoundException; +import java.io.FileOutputStream; +import java.io.FileWriter; import java.io.IOException; import java.net.DatagramPacket; import java.net.InetAddress; import java.net.SocketException; import java.net.SocketImpl; -import java.net.SocketOptions; + +import libcore.io.ErrnoException; import libcore.io.Libcore; import libcore.io.StructLinger; import org.apache.harmony.luni.platform.INetworkSystem; + import static libcore.io.OsConstants.*; /** @@ -44,6 +48,15 @@ import static libcore.io.OsConstants.*; */ public final class BlockGuard { + private static final boolean LOGI = true; + private static final boolean TAG_SOCKETS = false; + + // TODO: refactor class name to something more generic, since its scope is + // growing beyond just blocking/logging. + + private static final byte TAG_HEADER = 't'; + private static final byte TAG_SEPARATOR = '\0'; + public static final int DISALLOW_DISK_WRITE = 0x01; public static final int DISALLOW_DISK_READ = 0x02; public static final int DISALLOW_NETWORK = 0x04; @@ -77,6 +90,11 @@ public final class BlockGuard { int getPolicyMask(); } + public static class SocketTags { + public String statsTag = null; + public int statsUid = -1; + } + public static class BlockGuardPolicyException extends RuntimeException { // bitmask of DISALLOW_*, PENALTY_*, etc flags private final int mPolicyState; @@ -132,6 +150,13 @@ public final class BlockGuard { } }; + private static ThreadLocal<SocketTags> threadSocketTags = new ThreadLocal<SocketTags>() { + @Override + protected SocketTags initialValue() { + return new SocketTags(); + } + }; + /** * Get the current thread's policy. * @@ -142,6 +167,14 @@ public final class BlockGuard { return threadPolicy.get(); } + public static void setThreadSocketStatsTag(String tag) { + threadSocketTags.get().statsTag = tag; + } + + public static void setThreadSocketStatsUid(int uid) { + threadSocketTags.get().statsUid = uid; + } + /** * Sets the current thread's block guard policy. * @@ -155,6 +188,61 @@ public final class BlockGuard { threadPolicy.set(policy); } + public static void tagSocketFd(FileDescriptor fd) throws SocketException { + final SocketTags options = threadSocketTags.get(); + if (LOGI) { + System.logI("tagSocket(" + fd.getInt$() + ") with statsTag=" + + options.statsTag + ", statsUid=" + options.statsUid); + } + + try { + // TODO: skip tagging when options would be no-op + internalTagSocketFd(fd, options.statsTag, options.statsUid); + } catch (IOException e) { + throw new SocketException("Problem tagging socket", e); + } + } + + public static void untagSocketFd(FileDescriptor fd) throws SocketException { + if (LOGI) { + System.logI("untagSocket(" + fd.getInt$() + ")"); + } + + try { + internalTagSocketFd(fd, null, -1); + } catch (IOException e) { + throw new SocketException("Problem untagging socket", e); + } + } + + private static void internalTagSocketFd(FileDescriptor fd, String tag, int uid) + throws IOException { + if (!TAG_SOCKETS) return; + + final byte[] tagBytes = tag != null ? tag.getBytes() : new byte[0]; + final byte[] uidBytes = uid != -1 ? Integer.toString(uid).getBytes() : new byte[0]; + + final ByteArrayOutputStream buffer = new ByteArrayOutputStream( + 4 + tagBytes.length + uidBytes.length); + + buffer.write(TAG_HEADER); + buffer.write(TAG_SEPARATOR); + buffer.write(tagBytes); + buffer.write(TAG_SEPARATOR); + buffer.write(uidBytes); + buffer.write(TAG_SEPARATOR); + buffer.close(); + + final byte[] bufferBytes = buffer.toByteArray(); + + final FileOutputStream procOut = new FileOutputStream("/proc/net/qtaguid"); + try { + procOut.write(bufferBytes); + } finally { + procOut.close(); + } + } + private BlockGuard() {} /** @@ -171,6 +259,7 @@ public final class BlockGuard { FileDescriptor clientFd) throws IOException { BlockGuard.getThreadPolicy().onNetwork(); mNetwork.accept(serverFd, newSocket, clientFd); + tagSocketFd(clientFd); } public int read(FileDescriptor aFD, byte[] data, int offset, int count) throws IOException { @@ -235,7 +324,7 @@ public final class BlockGuard { mNetwork.close(aFD); } - private boolean isLingerSocket(FileDescriptor fd) throws SocketException { + private boolean isLingerSocket(FileDescriptor fd) { try { StructLinger linger = Libcore.os.getsockoptLinger(fd, SOL_SOCKET, SO_LINGER); return linger.isOn() && linger.l_linger > 0; |