• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.example.android.toyvpn;
18 
19 import android.app.Notification;
20 import android.app.PendingIntent;
21 import android.app.Service;
22 import android.content.Intent;
23 import android.content.SharedPreferences;
24 import android.net.VpnService;
25 import android.os.Handler;
26 import android.os.Message;
27 import android.os.ParcelFileDescriptor;
28 import android.util.Log;
29 import android.util.Pair;
30 import android.widget.Toast;
31 
32 import java.io.IOException;
33 import java.util.concurrent.atomic.AtomicInteger;
34 import java.util.concurrent.atomic.AtomicReference;
35 
36 public class ToyVpnService extends VpnService implements Handler.Callback {
37     private static final String TAG = ToyVpnService.class.getSimpleName();
38 
39     public static final String ACTION_CONNECT = "com.example.android.toyvpn.START";
40     public static final String ACTION_DISCONNECT = "com.example.android.toyvpn.STOP";
41 
42     private Handler mHandler;
43 
44     private static class Connection extends Pair<Thread, ParcelFileDescriptor> {
Connection(Thread thread, ParcelFileDescriptor pfd)45         public Connection(Thread thread, ParcelFileDescriptor pfd) {
46             super(thread, pfd);
47         }
48     }
49 
50     private final AtomicReference<Thread> mConnectingThread = new AtomicReference<>();
51     private final AtomicReference<Connection> mConnection = new AtomicReference<>();
52 
53     private AtomicInteger mNextConnectionId = new AtomicInteger(1);
54 
55     private PendingIntent mConfigureIntent;
56 
57     @Override
onCreate()58     public void onCreate() {
59         // The handler is only used to show messages.
60         if (mHandler == null) {
61             mHandler = new Handler(this);
62         }
63 
64         // Create the intent to "configure" the connection (just start ToyVpnClient).
65         mConfigureIntent = PendingIntent.getActivity(this, 0, new Intent(this, ToyVpnClient.class),
66                 PendingIntent.FLAG_UPDATE_CURRENT);
67     }
68 
69     @Override
onStartCommand(Intent intent, int flags, int startId)70     public int onStartCommand(Intent intent, int flags, int startId) {
71         if (intent != null && ACTION_DISCONNECT.equals(intent.getAction())) {
72             disconnect();
73             return START_NOT_STICKY;
74         } else {
75             connect();
76             return START_STICKY;
77         }
78     }
79 
80     @Override
onDestroy()81     public void onDestroy() {
82         disconnect();
83     }
84 
85     @Override
handleMessage(Message message)86     public boolean handleMessage(Message message) {
87         Toast.makeText(this, message.what, Toast.LENGTH_SHORT).show();
88         if (message.what != R.string.disconnected) {
89             updateForegroundNotification(message.what);
90         }
91         return true;
92     }
93 
connect()94     private void connect() {
95         // Become a foreground service. Background services can be VPN services too, but they can
96         // be killed by background check before getting a chance to receive onRevoke().
97         updateForegroundNotification(R.string.connecting);
98         mHandler.sendEmptyMessage(R.string.connecting);
99 
100         // Extract information from the shared preferences.
101         final SharedPreferences prefs = getSharedPreferences(ToyVpnClient.Prefs.NAME, MODE_PRIVATE);
102         final String server = prefs.getString(ToyVpnClient.Prefs.SERVER_ADDRESS, "");
103         final byte[] secret = prefs.getString(ToyVpnClient.Prefs.SHARED_SECRET, "").getBytes();
104         final int port;
105         try {
106             port = Integer.parseInt(prefs.getString(ToyVpnClient.Prefs.SERVER_PORT, ""));
107         } catch (NumberFormatException e) {
108             Log.e(TAG, "Bad port: " + prefs.getString(ToyVpnClient.Prefs.SERVER_PORT, null), e);
109             return;
110         }
111 
112         // Kick off a connection.
113         startConnection(new ToyVpnConnection(
114                 this, mNextConnectionId.getAndIncrement(), server, port, secret));
115     }
116 
startConnection(final ToyVpnConnection connection)117     private void startConnection(final ToyVpnConnection connection) {
118         // Replace any existing connecting thread with the  new one.
119         final Thread thread = new Thread(connection, "ToyVpnThread");
120         setConnectingThread(thread);
121 
122         // Handler to mark as connected once onEstablish is called.
123         connection.setConfigureIntent(mConfigureIntent);
124         connection.setOnEstablishListener(new ToyVpnConnection.OnEstablishListener() {
125             public void onEstablish(ParcelFileDescriptor tunInterface) {
126                 mHandler.sendEmptyMessage(R.string.connected);
127 
128                 mConnectingThread.compareAndSet(thread, null);
129                 setConnection(new Connection(thread, tunInterface));
130             }
131         });
132         thread.start();
133     }
134 
setConnectingThread(final Thread thread)135     private void setConnectingThread(final Thread thread) {
136         final Thread oldThread = mConnectingThread.getAndSet(thread);
137         if (oldThread != null) {
138             oldThread.interrupt();
139         }
140     }
141 
setConnection(final Connection connection)142     private void setConnection(final Connection connection) {
143         final Connection oldConnection = mConnection.getAndSet(connection);
144         if (oldConnection != null) {
145             try {
146                 oldConnection.first.interrupt();
147                 oldConnection.second.close();
148             } catch (IOException e) {
149                 Log.e(TAG, "Closing VPN interface", e);
150             }
151         }
152     }
153 
disconnect()154     private void disconnect() {
155         mHandler.sendEmptyMessage(R.string.disconnected);
156         setConnectingThread(null);
157         setConnection(null);
158         stopForeground(true);
159     }
160 
updateForegroundNotification(final int message)161     private void updateForegroundNotification(final int message) {
162         startForeground(1, new Notification.Builder(this)
163                 .setSmallIcon(R.drawable.ic_vpn)
164                 .setContentText(getString(message))
165                 .setContentIntent(mConfigureIntent)
166                 .build());
167     }
168 }
169