Home | History | Annotate | Download | only in location
      1 /*
      2  * Copyright (C) 2008 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 android.location;
     18 
     19 import java.util.Iterator;
     20 import java.util.NoSuchElementException;
     21 
     22 
     23 /**
     24  * This class represents the current state of the GPS engine.
     25  * This class is used in conjunction with the {@link Listener} interface.
     26  */
     27 public final class GpsStatus {
     28     private static final int NUM_SATELLITES = 255;
     29 
     30     /* These package private values are modified by the LocationManager class */
     31     private int mTimeToFirstFix;
     32     private GpsSatellite mSatellites[] = new GpsSatellite[NUM_SATELLITES];
     33 
     34     private final class SatelliteIterator implements Iterator<GpsSatellite> {
     35 
     36         private GpsSatellite[] mSatellites;
     37         int mIndex = 0;
     38 
     39         SatelliteIterator(GpsSatellite[] satellites) {
     40             mSatellites = satellites;
     41         }
     42 
     43         public boolean hasNext() {
     44             for (int i = mIndex; i < mSatellites.length; i++) {
     45                 if (mSatellites[i].mValid) {
     46                     return true;
     47                 }
     48             }
     49             return false;
     50         }
     51 
     52         public GpsSatellite next() {
     53             while (mIndex < mSatellites.length) {
     54                 GpsSatellite satellite = mSatellites[mIndex++];
     55                 if (satellite.mValid) {
     56                     return satellite;
     57                 }
     58             }
     59             throw new NoSuchElementException();
     60         }
     61 
     62         public void remove() {
     63             throw new UnsupportedOperationException();
     64         }
     65     }
     66 
     67     private Iterable<GpsSatellite> mSatelliteList = new Iterable<GpsSatellite>() {
     68         public Iterator<GpsSatellite> iterator() {
     69             return new SatelliteIterator(mSatellites);
     70         }
     71     };
     72 
     73     /**
     74      * Event sent when the GPS system has started.
     75      */
     76     public static final int GPS_EVENT_STARTED = 1;
     77 
     78     /**
     79      * Event sent when the GPS system has stopped.
     80      */
     81     public static final int GPS_EVENT_STOPPED = 2;
     82 
     83     /**
     84      * Event sent when the GPS system has received its first fix since starting.
     85      * Call {@link #getTimeToFirstFix()} to find the time from start to first fix.
     86      */
     87     public static final int GPS_EVENT_FIRST_FIX = 3;
     88 
     89     /**
     90      * Event sent periodically to report GPS satellite status.
     91      * Call {@link #getSatellites()} to retrieve the status for each satellite.
     92      */
     93     public static final int GPS_EVENT_SATELLITE_STATUS = 4;
     94 
     95     /**
     96      * Used for receiving notifications when GPS status has changed.
     97      */
     98     public interface Listener {
     99         /**
    100          * Called to report changes in the GPS status.
    101          * The event number is one of:
    102          * <ul>
    103          * <li> {@link GpsStatus#GPS_EVENT_STARTED}
    104          * <li> {@link GpsStatus#GPS_EVENT_STOPPED}
    105          * <li> {@link GpsStatus#GPS_EVENT_FIRST_FIX}
    106          * <li> {@link GpsStatus#GPS_EVENT_SATELLITE_STATUS}
    107          * </ul>
    108          *
    109          * When this method is called, the client should call
    110          * {@link LocationManager#getGpsStatus} to get additional
    111          * status information.
    112          *
    113          * @param event event number for this notification
    114          */
    115         void onGpsStatusChanged(int event);
    116     }
    117 
    118     /**
    119      * Used for receiving NMEA sentences from the GPS.
    120      * NMEA 0183 is a standard for communicating with marine electronic devices
    121      * and is a common method for receiving data from a GPS, typically over a serial port.
    122      * See <a href="http://en.wikipedia.org/wiki/NMEA_0183">NMEA 0183</a> for more details.
    123      * You can implement this interface and call {@link LocationManager#addNmeaListener}
    124      * to receive NMEA data from the GPS engine.
    125      */
    126     public interface NmeaListener {
    127         void onNmeaReceived(long timestamp, String nmea);
    128     }
    129 
    130     GpsStatus() {
    131         for (int i = 0; i < mSatellites.length; i++) {
    132             mSatellites[i] = new GpsSatellite(i + 1);
    133         }
    134     }
    135 
    136     /**
    137      * Used internally within {@link LocationManager} to copy GPS status
    138      * data from the Location Manager Service to its cached GpsStatus instance.
    139      * Is synchronized to ensure that GPS status updates are atomic.
    140      */
    141     synchronized void setStatus(int svCount, int[] prns, float[] snrs,
    142             float[] elevations, float[] azimuths, int ephemerisMask,
    143             int almanacMask, int usedInFixMask) {
    144         int i;
    145 
    146         for (i = 0; i < mSatellites.length; i++) {
    147             mSatellites[i].mValid = false;
    148         }
    149 
    150         for (i = 0; i < svCount; i++) {
    151             int prn = prns[i] - 1;
    152             int prnShift = (1 << prn);
    153             if (prn >= 0 && prn < mSatellites.length) {
    154                 GpsSatellite satellite = mSatellites[prn];
    155 
    156                 satellite.mValid = true;
    157                 satellite.mSnr = snrs[i];
    158                 satellite.mElevation = elevations[i];
    159                 satellite.mAzimuth = azimuths[i];
    160                 satellite.mHasEphemeris = ((ephemerisMask & prnShift) != 0);
    161                 satellite.mHasAlmanac = ((almanacMask & prnShift) != 0);
    162                 satellite.mUsedInFix = ((usedInFixMask & prnShift) != 0);
    163             }
    164         }
    165     }
    166 
    167     /**
    168      * Used by {@link LocationManager#getGpsStatus} to copy LocationManager's
    169      * cached GpsStatus instance to the client's copy.
    170      * Since this method is only used within {@link LocationManager#getGpsStatus},
    171      * it does not need to be synchronized.
    172      */
    173     void setStatus(GpsStatus status) {
    174         mTimeToFirstFix = status.getTimeToFirstFix();
    175 
    176         for (int i = 0; i < mSatellites.length; i++) {
    177             mSatellites[i].setStatus(status.mSatellites[i]);
    178         }
    179     }
    180 
    181     void setTimeToFirstFix(int ttff) {
    182         mTimeToFirstFix = ttff;
    183     }
    184 
    185     /**
    186      * Returns the time required to receive the first fix since the most recent
    187      * restart of the GPS engine.
    188      *
    189      * @return time to first fix in milliseconds
    190      */
    191     public int getTimeToFirstFix() {
    192         return mTimeToFirstFix;
    193     }
    194 
    195     /**
    196      * Returns an array of {@link GpsSatellite} objects, which represent the
    197      * current state of the GPS engine.
    198      *
    199      * @return the list of satellites
    200      */
    201     public Iterable<GpsSatellite> getSatellites() {
    202         return mSatelliteList;
    203     }
    204 
    205     /**
    206      * Returns the maximum number of satellites that can be in the satellite
    207      * list that can be returned by {@link #getSatellites()}.
    208      *
    209      * @return the maximum number of satellites
    210      */
    211     public int getMaxSatellites() {
    212         return NUM_SATELLITES;
    213     }
    214 }
    215