Skip to content

Commit

Permalink
lib: location: skip A-GNSS data request when no ephe/alm requested
Browse files Browse the repository at this point in the history
GNSS asks for some A-GNSS data, like UTC and iono parameters, once every
24 hours. This may have caused a separate A-GNSS data request for this
data alone. The implementation has now been modified to skip such a
request. All data is downloaded with the next ephemeris request, which
happens approximately every two hours.

Signed-off-by: Tommi Kangas <[email protected]>
  • Loading branch information
tokangas committed Jan 16, 2025
1 parent c9251c2 commit 5a73123
Show file tree
Hide file tree
Showing 2 changed files with 194 additions and 48 deletions.
30 changes: 17 additions & 13 deletions lib/location/method_gnss.c
Original file line number Diff line number Diff line change
Expand Up @@ -425,20 +425,28 @@ static void method_gnss_pgps_request_work_fn(struct k_work *item)
static bool method_gnss_agnss_required(void)
{
int32_t time_since_agnss_req;
bool ephe_or_alm_required = false;
bool ephe_requested = false;
bool alm_requested = false;

/* Check if A-GNSS data is needed. */
for (int i = 0; i < agnss_request.system_count; i++) {
if (agnss_request.system[i].sv_mask_ephe != 0 ||
agnss_request.system[i].sv_mask_alm != 0) {
ephe_or_alm_required = true;
break;
if (agnss_request.system[i].sv_mask_ephe != 0) {
ephe_requested = true;
}
if (agnss_request.system[i].sv_mask_alm != 0) {
alm_requested = true;
}
}

if (agnss_request.data_flags == 0 && !ephe_or_alm_required) {
if (!ephe_requested && !alm_requested && agnss_request.data_flags == 0) {
LOG_DBG("No A-GNSS data types requested");
return false;
} else if (!IS_ENABLED(CONFIG_NRF_CLOUD_PGPS) && !ephe_requested) {
/* No ephemerides requested and A-GNSS is used to provide ephemerides.
* Skip this request and download all data with the next ephemeris request.
*/
LOG_DBG("Skipping A-GNSS request, no ephemerides requested");
return false;
}

/* A-GNSS data is needed, check if enough time has passed since the last A-GNSS data
Expand Down Expand Up @@ -551,12 +559,8 @@ static void method_gnss_assistance_request(void)
* QZSS satellites always as expired.
*/
if (agnss_request.system_count > 1) {
if (agnss_request.data_flags != 0 ||
agnss_request.system[0].sv_mask_ephe != 0 ||
agnss_request.system[0].sv_mask_alm != 0) {
/* QZSS ephemerides are requested always when other assistance data is
* needed.
*/
if (agnss_request.system[0].sv_mask_ephe != 0) {
/* QZSS ephemerides are requested whenever GPS ephemerides are requested. */
agnss_request.system[1].sv_mask_ephe = 0x3ff;
} else {
/* No other assistance is needed. Request QZSS ephemerides anyway if
Expand Down Expand Up @@ -1172,7 +1176,7 @@ static void method_gnss_agnss_expiry_process(const struct nrf_modem_gnss_agnss_e
*/

/* QZSS ephemerides are valid for a maximum of two hours, so no expiry
* is used here.
* threshold is used here.
*/
if (agnss_expiry->sv[i].ephe_expiry == 0) {
expired_qzss_ephe_mask |=
Expand Down
212 changes: 177 additions & 35 deletions tests/lib/location/src/location_test.c
Original file line number Diff line number Diff line change
Expand Up @@ -576,6 +576,156 @@ int nrf_cloud_rest_agnss_data_get_Stub(
return 0;
}

/* Test that no A-GNSS data is requested when less than 3 GPS satellites have expired
* ephemerides, even when all other data is expired.
*/
void test_location_gnss_agnss_no_request(void)
{
int err;
struct location_config config = { 0 };
enum location_method methods[] = {LOCATION_METHOD_GNSS};

#if !defined(CONFIG_LOCATION_TEST_AGNSS)
TEST_IGNORE();
#endif

location_config_defaults_set(&config, 1, methods);

#if defined(CONFIG_LOCATION_DATA_DETAILS)
test_location_event_data[location_cb_expected].id = LOCATION_EVT_STARTED;
test_location_event_data[location_cb_expected].method = LOCATION_METHOD_GNSS;
location_cb_expected++;
#endif

#if defined(CONFIG_LOCATION_SERVICE_EXTERNAL)
test_location_event_data[location_cb_expected].id = LOCATION_EVT_GNSS_ASSISTANCE_REQUEST;
test_location_event_data[location_cb_expected].method = LOCATION_METHOD_GNSS;
location_cb_expected++;
#endif

test_pvt_data.flags = NRF_MODEM_GNSS_PVT_FLAG_FIX_VALID;
test_pvt_data.latitude = 61.005;
test_pvt_data.longitude = -45.997;
test_pvt_data.accuracy = 15.83;
test_pvt_data.datetime.year = 2021;
test_pvt_data.datetime.month = 8;
test_pvt_data.datetime.day = 13;
test_pvt_data.datetime.hour = 12;
test_pvt_data.datetime.minute = 34;
test_pvt_data.datetime.seconds = 56;
test_pvt_data.datetime.ms = 789;
test_pvt_data.sv[0].sv = 2;
test_pvt_data.sv[0].flags = NRF_MODEM_GNSS_SV_FLAG_USED_IN_FIX;
test_pvt_data.sv[1].sv = 4;
test_pvt_data.sv[1].flags = NRF_MODEM_GNSS_SV_FLAG_USED_IN_FIX;
test_pvt_data.sv[2].sv = 6;
test_pvt_data.sv[2].flags = NRF_MODEM_GNSS_SV_FLAG_USED_IN_FIX;
test_pvt_data.sv[3].sv = 8;
test_pvt_data.sv[3].flags = NRF_MODEM_GNSS_SV_FLAG_USED_IN_FIX;
test_pvt_data.sv[4].sv = 10;
test_pvt_data.sv[4].flags = NRF_MODEM_GNSS_SV_FLAG_USED_IN_FIX;

test_location_event_data[location_cb_expected].id = LOCATION_EVT_LOCATION;
test_location_event_data[location_cb_expected].method = LOCATION_METHOD_GNSS;
test_location_event_data[location_cb_expected].location.latitude = 61.005;
test_location_event_data[location_cb_expected].location.longitude = -45.997;
test_location_event_data[location_cb_expected].location.accuracy = 15.83;
test_location_event_data[location_cb_expected].location.datetime.valid = true;
test_location_event_data[location_cb_expected].location.datetime.year = 2021;
test_location_event_data[location_cb_expected].location.datetime.month = 8;
test_location_event_data[location_cb_expected].location.datetime.day = 13;
test_location_event_data[location_cb_expected].location.datetime.hour = 12;
test_location_event_data[location_cb_expected].location.datetime.minute = 34;
test_location_event_data[location_cb_expected].location.datetime.second = 56;
test_location_event_data[location_cb_expected].location.datetime.ms = 789;
#if defined(CONFIG_LOCATION_DATA_DETAILS)
test_location_event_data[location_cb_expected].location.details.gnss.satellites_tracked = 5;
test_location_event_data[location_cb_expected].location.details.gnss.satellites_used = 5;
test_location_event_data[location_cb_expected].location.details.gnss.elapsed_time_gnss = 50;
test_location_event_data[location_cb_expected].location.details.gnss.pvt_data =
test_pvt_data;
#endif
location_cb_expected++;

__cmock_nrf_modem_gnss_event_handler_set_ExpectAndReturn(&method_gnss_event_handler, 0);

/* The expiry times are in minutes. Most of the data types are considered expired when
* there's less than 80 minutes left.
*/
struct nrf_modem_gnss_agnss_expiry agnss_expiry = {
/* All generic assistance data is expired. */
.data_flags =
NRF_MODEM_GNSS_AGNSS_GPS_UTC_REQUEST |
NRF_MODEM_GNSS_AGNSS_KLOBUCHAR_REQUEST |
NRF_MODEM_GNSS_AGNSS_NEQUICK_REQUEST |
NRF_MODEM_GNSS_AGNSS_POSITION_REQUEST |
NRF_MODEM_GNSS_AGNSS_INTEGRITY_REQUEST,
.utc_expiry = 50, /* expired */
.klob_expiry = 50, /* expired */
.neq_expiry = 50, /* expired */
.integrity_expiry = 50, /* expired */
.position_expiry = 0, /* expired, no expiry threshold used for position */
.sv_count = 32 + 10 /* expired */
};

/* Two GPS satellites have expired ephemerides, all GPS satellites have expired almanacs. */
for (int i = 0; i < 32; i++) {
agnss_expiry.sv[i].sv_id = i + 1;
agnss_expiry.sv[i].system_id = NRF_MODEM_GNSS_SYSTEM_GPS;
agnss_expiry.sv[i].ephe_expiry = 120; /* valid */
agnss_expiry.sv[i].alm_expiry = 50; /* expired */
}
agnss_expiry.sv[0].ephe_expiry = 50; /* expired */
agnss_expiry.sv[1].ephe_expiry = 50; /* expired */

/* All QZSS satellites have expired ephemerides and almanacs. */
for (int i = 0; i < 10; i++) {
agnss_expiry.sv[i + 32].sv_id = i + 1;
agnss_expiry.sv[i + 32].system_id = NRF_MODEM_GNSS_SYSTEM_QZSS;
/* No expiry threshold used for QZSS. */
agnss_expiry.sv[i + 32].ephe_expiry = 0; /* expired */
agnss_expiry.sv[i + 32].alm_expiry = 0; /* expired */
}

__cmock_nrf_modem_gnss_agnss_expiry_get_ExpectAndReturn(NULL, 0);
__cmock_nrf_modem_gnss_agnss_expiry_get_IgnoreArg_agnss_expiry();
__cmock_nrf_modem_gnss_agnss_expiry_get_ReturnMemThruPtr_agnss_expiry(
&agnss_expiry, sizeof(agnss_expiry));

__cmock_nrf_modem_gnss_fix_interval_set_ExpectAndReturn(1, 0);
__cmock_nrf_modem_gnss_use_case_set_ExpectAndReturn(
NRF_MODEM_GNSS_USE_CASE_MULTIPLE_HOT_START, 0);
__cmock_nrf_modem_gnss_start_ExpectAndReturn(0);

__mock_nrf_modem_at_scanf_ExpectAndReturn(
"AT%XSYSTEMMODE?", "%%XSYSTEMMODE: %d,%d,%d,%d", 4);
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* LTE-M support */
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* NB-IoT support */
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* GNSS support */
__mock_nrf_modem_at_scanf_ReturnVarg_int(0); /* LTE preference */

err = location_request(&config);
TEST_ASSERT_EQUAL(0, err);
k_sleep(K_MSEC(1));

#if defined(CONFIG_LOCATION_DATA_DETAILS)
err = k_sem_take(&event_handler_called_sem, K_SECONDS(3));
TEST_ASSERT_EQUAL(0, err);
#endif

/* An event other than NRF_MODEM_GNSS_EVT_PVT to see that no actions are done */
method_gnss_event_handler(NRF_MODEM_GNSS_EVT_FIX);
k_sleep(K_MSEC(1));

__cmock_nrf_modem_gnss_read_ExpectAndReturn(
NULL, sizeof(test_pvt_data), NRF_MODEM_GNSS_DATA_PVT, 0);
__cmock_nrf_modem_gnss_read_IgnoreArg_buf();
__cmock_nrf_modem_gnss_read_ReturnMemThruPtr_buf(&test_pvt_data, sizeof(test_pvt_data));
__cmock_nrf_modem_gnss_stop_ExpectAndReturn(0);
method_gnss_event_handler(NRF_MODEM_GNSS_EVT_PVT);
k_sleep(K_MSEC(1));
}

/* Test successful GNSS location request. */
void test_location_gnss(void)
{
Expand Down Expand Up @@ -645,51 +795,49 @@ void test_location_gnss(void)
__cmock_nrf_modem_gnss_event_handler_set_ExpectAndReturn(&method_gnss_event_handler, 0);

#if defined(CONFIG_LOCATION_TEST_AGNSS)
/* Zero triggers new AGNSS data request */
/* The expiry times are in minutes. Most of the data types are considered expired when
* there's less than 80 minutes left.
*/
struct nrf_modem_gnss_agnss_expiry agnss_expiry = {
.data_flags = NRF_MODEM_GNSS_AGNSS_GPS_SYS_TIME_AND_SV_TOW_REQUEST,
.utc_expiry = 1,
.klob_expiry = 2,
.neq_expiry = 3,
.integrity_expiry = 4,
.position_expiry = 5,
.sv_count = 6,
/* Only position is expired, but all generic assistance data should be
* requested at the same time.
*/
.data_flags = NRF_MODEM_GNSS_AGNSS_POSITION_REQUEST,
.utc_expiry = 120, /* valid */
.klob_expiry = 120, /* valid */
.neq_expiry = 120, /* valid */
.integrity_expiry = 120, /* valid */
.position_expiry = 0, /* expired */
.sv_count = 4,
.sv = {
/* Three expired GPS ephes and almanacs are needed to trigger a request. */
{
.sv_id = 1,
.system_id = NRF_MODEM_GNSS_SYSTEM_GPS,
.ephe_expiry = 1,
.alm_expiry = 4
.ephe_expiry = 50, /* expired */
.alm_expiry = 50 /* expired */
},
{
.sv_id = 2,
.system_id = NRF_MODEM_GNSS_SYSTEM_GPS,
.ephe_expiry = 100,
.alm_expiry = 4
.ephe_expiry = 50, /* expired */
.alm_expiry = 50 /* expired */
},
{
.sv_id = 3,
.system_id = NRF_MODEM_GNSS_SYSTEM_GPS,
.ephe_expiry = 0,
.alm_expiry = 101
},
{
.sv_id = 4,
.system_id = NRF_MODEM_GNSS_SYSTEM_GPS,
.ephe_expiry = 6,
.alm_expiry = 1
.ephe_expiry = 50, /* expired */
.alm_expiry = 50 /* expired */
},
/* QZSS ephes and almanacs are requested at the same time with GPS, when
* QZSS is supported. It's enough to have one QZSS SV and data doesn't
* even have to be expired.
*/
{
.sv_id = 193,
.system_id = NRF_MODEM_GNSS_SYSTEM_QZSS,
.ephe_expiry = 0,
.alm_expiry = 100
},
{
.sv_id = 202,
.system_id = NRF_MODEM_GNSS_SYSTEM_QZSS,
.ephe_expiry = 100,
.alm_expiry = 4
.ephe_expiry = 120, /* valid */
.alm_expiry = 120 /* valid */
},
}
};
Expand Down Expand Up @@ -740,7 +888,7 @@ void test_location_gnss(void)
test_agnss_data, sizeof(test_agnss_data), 0);
#endif
#endif
/* TODO: Cannot determine the used system mode but it's set as zero by default in lte_lc */

__mock_nrf_modem_at_scanf_ExpectAndReturn(
"AT%XSYSTEMMODE?", "%%XSYSTEMMODE: %d,%d,%d,%d", 4);
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* LTE-M support */
Expand Down Expand Up @@ -842,7 +990,6 @@ void test_location_gnss_location_request_timeout(void)
NRF_MODEM_GNSS_USE_CASE_MULTIPLE_HOT_START, 0);
__cmock_nrf_modem_gnss_start_ExpectAndReturn(0);

/* TODO: Cannot determine the used system mode but it's set as zero by default in lte_lc */
__mock_nrf_modem_at_scanf_ExpectAndReturn(
"AT%XSYSTEMMODE?", "%%XSYSTEMMODE: %d,%d,%d,%d", 4);
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* LTE-M support */
Expand Down Expand Up @@ -1452,7 +1599,6 @@ void test_location_request_default(void)
NRF_MODEM_GNSS_USE_CASE_MULTIPLE_HOT_START, 0);
__cmock_nrf_modem_gnss_start_ExpectAndReturn(0);

/* TODO: Cannot determine the used system mode but it's set as zero by default in lte_lc */
__mock_nrf_modem_at_scanf_ExpectAndReturn(
"AT%XSYSTEMMODE?", "%%XSYSTEMMODE: %d,%d,%d,%d", 4);
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* LTE-M support */
Expand Down Expand Up @@ -1684,7 +1830,6 @@ void test_location_request_mode_all_cellular_gnss(void)
NRF_MODEM_GNSS_USE_CASE_MULTIPLE_HOT_START, 0);
__cmock_nrf_modem_gnss_start_ExpectAndReturn(0);

/* TODO: Cannot determine the used system mode but it's set as zero by default in lte_lc */
__mock_nrf_modem_at_scanf_ExpectAndReturn(
"AT%XSYSTEMMODE?", "%%XSYSTEMMODE: %d,%d,%d,%d", 4);
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* LTE-M support */
Expand Down Expand Up @@ -1786,7 +1931,6 @@ void test_location_request_mode_all_cellular_error_gnss_timeout(void)
NRF_MODEM_GNSS_USE_CASE_MULTIPLE_HOT_START, 0);
__cmock_nrf_modem_gnss_start_ExpectAndReturn(0);

/* TODO: Cannot determine the used system mode but it's set as zero by default in lte_lc */
__mock_nrf_modem_at_scanf_ExpectAndReturn(
"AT%XSYSTEMMODE?", "%%XSYSTEMMODE: %d,%d,%d,%d", 4);
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* LTE-M support */
Expand Down Expand Up @@ -1885,7 +2029,6 @@ void test_location_gnss_periodic(void)
NRF_MODEM_GNSS_USE_CASE_MULTIPLE_HOT_START, 0);
__cmock_nrf_modem_gnss_start_ExpectAndReturn(0);

/* TODO: Cannot determine the used system mode but it's set as zero by default in lte_lc */
__mock_nrf_modem_at_scanf_ExpectAndReturn(
"AT%XSYSTEMMODE?", "%%XSYSTEMMODE: %d,%d,%d,%d", 4);
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* LTE-M support */
Expand Down Expand Up @@ -1947,7 +2090,6 @@ void test_location_gnss_periodic(void)
NRF_MODEM_GNSS_USE_CASE_MULTIPLE_HOT_START, 0);
__cmock_nrf_modem_gnss_start_ExpectAndReturn(0);

/* TODO: Cannot determine the used system mode but it's set as zero by default in lte_lc */
__mock_nrf_modem_at_scanf_ExpectAndReturn(
"AT%XSYSTEMMODE?", "%%XSYSTEMMODE: %d,%d,%d,%d", 4);
__mock_nrf_modem_at_scanf_ReturnVarg_int(1); /* LTE-M support */
Expand Down

0 comments on commit 5a73123

Please sign in to comment.