mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
Update the README.md
Fix the m_rb bug in bt_app_hf.c Change the log level into information.
This commit is contained in:
parent
88c8791b1d
commit
742cb8df9e
@ -248,7 +248,7 @@ typedef union {
|
||||
const char *number; /*!< phone number corresponding to the last voice tag in the HF */
|
||||
} binp; /*!< HF callback param of ESP_HF_CLIENT_BINP_EVT */
|
||||
|
||||
} esp_hf_client_cb_param_t;
|
||||
} esp_hf_client_cb_param_t; /*!< HFP client callback parameters */
|
||||
|
||||
/**
|
||||
* @brief HFP client incoming data callback function, the callback is useful in case of
|
||||
|
@ -228,15 +228,14 @@ static void hci_update_adv_report_flow_control(BT_HDR *packet)
|
||||
// update adv free number
|
||||
hci_hal_env.adv_free_num ++;
|
||||
if (esp_vhci_host_check_send_available()){
|
||||
#if (BLE_INCLUDED == TRUE)
|
||||
// send hci cmd
|
||||
btsnd_hcic_ble_update_adv_report_flow_control(hci_hal_env.adv_free_num);
|
||||
#endif
|
||||
hci_hal_env.adv_free_num = 0;
|
||||
} else {
|
||||
//do nothing
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
#endif
|
||||
|
||||
|
@ -1,12 +1,12 @@
|
||||
# Hands-Free Unit
|
||||
|
||||
This example is to show how to use the APIs of Hands-Free (HF) Unit Component and the effects of them with the help of structured commands. You can use this example to communicate with an Audio Gateway (AG) device (e.g. a smart phone). This example use the UART as a transportation of commands and reserve the GPIO for PCM audio path.
|
||||
This example is to show how to use the APIs of Hands-Free (HF) Unit Component and the effects of them by providing a set of commands. You can use this example to communicate with an Audio Gateway (AG) device (e.g. a smart phone). This example uses UART for user commands.
|
||||
|
||||
## How to use example
|
||||
|
||||
### Hardware Required
|
||||
|
||||
If possible, example should be able to run on any commonly available ESP32 development board. And is supposed to connect to _Hands Free Audio Gateway (hfp_ag)_ example in ESP-IDF.
|
||||
This example is designed to run on commonly available ESP32 development board, e.g. ESP32-DevKitC. To operate it should be connected to an AG running on a smartphone or on another ESP32 development board loaded with Hands Free Audio Gateway (hfp_ag) example from ESP-IDF.
|
||||
|
||||
### Configure the project
|
||||
|
||||
@ -14,9 +14,12 @@ If possible, example should be able to run on any commonly available ESP32 devel
|
||||
idf.py menuconfig
|
||||
```
|
||||
|
||||
- Enable `Classic Bluetooth`, `Hands Free/Handset` and `Hands Free Unit` under `Component config ---> Bluetooth ---> Bluedroid Options`
|
||||
- When using PCM as the data path and this example configures PCM audio data to GPIO pins. You can link the GPIO pins to a speaker via i2s port. PCM data path does not support mSBC codec but CVSD codec.
|
||||
- When using HCI data path, ESP32 support both CVSD and mSBC codec.
|
||||
ESP32 supports two types of audio data path: PCM() and HCI(Host-Controller Interface) but the default sdkconfig of this example does not config the data path in a specific way. You should config the data path you want:
|
||||
|
||||
- PCM : When using PCM, audio data stream is mapped to GPIO pins and you should link these GPIO pins to a speaker via I2S port. And you should choose PCM in `menuconfig` path: `Component config --> Bluetooth controller --> BR/EDR Sync(SCO/eSCO) default data path --> PCM`and also `Component config --> Bluetooth --> Bluedroid Options -->Hands Free/Handset Profile --> audio(SCO) data path --> PCM`.
|
||||
- HCI : When using HCI, audio data stream will be transport to HF unit app via HCI. And you should choose HCI in `menuconfig` path: `Component config -->Bluetooth controller -->BR/EDR Sync(SCO/eSCO) default data path --> HCI` and also `Component config --> Bluetooth --> Bluedroid Options -->Hands Free/Handset Profile --> audio(SCO) data path --> HCI`.
|
||||
|
||||
**Note: Wide Band Speech is disabled by default, if you want to use it please select it in menuconfig path: `Component config --> Bluetooth --> Bluedroid Options --> Wide Band Speech`.**
|
||||
|
||||
### Build and Flash
|
||||
|
||||
@ -30,16 +33,16 @@ idf.py -p PORT flash monitor
|
||||
|
||||
(To exit the serial monitor, type ``Ctrl-]``.)
|
||||
|
||||
See the Getting Started Guide for full steps to configure and use ESP-IDF to build projects.
|
||||
See the [Getting Started Guide](https://docs.espressif.com/projects/esp-idf/en/latest/get-started/index.html) for full steps to configure and use ESP-IDF to build projects.
|
||||
|
||||
## Example Output
|
||||
|
||||
When you run this example, the command help table will print out in the monitor:
|
||||
When you run this example, the commands help table prints the following at the very begining:
|
||||
|
||||
```
|
||||
########################################################################
|
||||
HF client command usage manual
|
||||
HF client commands begin with "hf" and ends with ";"
|
||||
HF client commands begin with "hf" and end with ";"
|
||||
Supported commands are as follows, arguments are embraced with < and >
|
||||
hf con; -- setup connection with peer device
|
||||
hf dis; -- release connection with peer device
|
||||
@ -49,80 +52,114 @@ hf qop; -- query current operator name
|
||||
hf qc; -- query current call status
|
||||
hf ac; -- answer incoming call
|
||||
hf rc; -- reject incoming call
|
||||
hf d <num>; -- dial <num>, e.g. hf d 11223344
|
||||
hf rd; -- redial
|
||||
hf dm <index>; -- dial memory
|
||||
hf vron; -- start voice recognition
|
||||
hf vroff; -- stop voice recognition
|
||||
hf vu <tgt> <vol>; -- volume update
|
||||
tgt: 0-speaker, 1-microphone
|
||||
vol: volume gain ranges from 0 to 15
|
||||
hf rs; -- retrieve subscriber information
|
||||
hf rv; -- retrieve last voice tag number
|
||||
hf rh <btrh>; -- response and hold
|
||||
btrh:
|
||||
0 - put call on hold,
|
||||
1 - accept the held call,
|
||||
2 -reject the held call
|
||||
hf k <dtmf>; -- send dtmf code.
|
||||
dtmf: single character in set 0-9, *, #, A-D
|
||||
hf h; -- show command manual
|
||||
hf d <num>; -- dial <num>, e.g. hf d 11223344
|
||||
hf rd; -- redial
|
||||
hf dm <index>; -- dial memory
|
||||
hf vron; -- start voice recognition
|
||||
hf vroff; -- stop voice recognition
|
||||
hf vu <tgt> <vol>; -- volume update
|
||||
tgt: 0-speaker, 1-microphone
|
||||
vol: volume gain ranges from 0 to 15
|
||||
hf rs; -- retrieve subscriber information
|
||||
hf rv; -- retrieve last voice tag number
|
||||
hf rh <btrh>; -- response and hold
|
||||
btrh:
|
||||
0 - put call on hold,
|
||||
1 - accept the held call,
|
||||
2 -reject the held call
|
||||
hf k <dtmf>; -- send dtmf code.
|
||||
dtmf: single character in set 0-9, *, #, A-D
|
||||
hf h; -- show command manual
|
||||
########################################################################
|
||||
```
|
||||
|
||||
The commands help table will print out in monitor whenever you type `hf h;` or input a command that is not required by the command parse rule.
|
||||
**Note:**
|
||||
|
||||
- This command help table will print out in monitor whenever you type `hf h;` or if you input a command that is not required by the command parse rule.
|
||||
- The command you type will not echo in monitor and your command should always start with `hf` and end with `;` or the example will not responds.
|
||||
- The command you typed will not echo in monitor.
|
||||
|
||||
### Service Level Connection and Disconnection
|
||||
|
||||
In this example, type `hf con;` will trigger the HF Unit to establish a service level connection with AG device and log prints like:
|
||||
You can type `hf con;` to establish a service level connection with AG device and log prints such as:
|
||||
|
||||
```
|
||||
E (100147) CNSL: Command [hf dis;]
|
||||
E (78502) CNSL: Command [hf con;]
|
||||
connect
|
||||
W (79632) BT_APPL: new conn_srvc id:27, app_id:1
|
||||
I (79642) BT_HF: APP HFP event: CONNECTION_STATE_EVT
|
||||
I (79642) BT_HF: --connection state connected, peer feats 0x0, chld_feats 0x0
|
||||
I (79792) BT_HF: APP HFP event: CALL_IND_EVT
|
||||
I (79792) BT_HF: --Call indicator NO call in progress
|
||||
I (79792) BT_HF: APP HFP event: CALL_SETUP_IND_EVT
|
||||
I (79802) BT_HF: --Call setup indicator NONE
|
||||
I (79802) BT_HF: APP HFP event: NETWORK_STATE_EVT
|
||||
I (79812) BT_HF: --NETWORK STATE available
|
||||
I (79812) BT_HF: APP HFP event: SIGNAL_STRENGTH_IND_EVT
|
||||
I (79822) BT_HF: -- signal strength: 4
|
||||
I (79822) BT_HF: APP HFP event: ROAMING_STATUS_IND_EVT
|
||||
I (79832) BT_HF: --ROAMING: inactive
|
||||
I (79832) BT_HF: APP HFP event: BATTERY_LEVEL_IND_EVT
|
||||
I (79842) BT_HF: --battery level 3
|
||||
I (79842) BT_HF: APP HFP event: CALL_HELD_IND_EVT
|
||||
I (79852) BT_HF: --Call held indicator NONE held
|
||||
I (79852) BT_HF: APP HFP event: CONNECTION_STATE_EVT
|
||||
I (79862) BT_HF: --connection state slc_connected, peer feats 0x16e, chld_feats 0x0
|
||||
I (79872) BT_HF: APP HFP event: INBAND_RING_TONE_EVT
|
||||
I (79872) BT_HF: --inband ring state Provided
|
||||
```
|
||||
|
||||
**Note: Only after Hands-Free Profile(HFP) service is initialized and a service level connection exists between an HF Unit and an AG device, could other commands be available.**
|
||||
|
||||
You can type `hf dis;` to disconnect with the connected AG device, and log prints such as:
|
||||
|
||||
```
|
||||
E (93382) CNSL: Command [hf dis;]
|
||||
disconnect
|
||||
W (100427) BT_RFCOMM: port_rfc_closed RFCOMM connection in state 3 closed: Closed (res: 19)
|
||||
W (100427) BT_APPL: BTA_HF_CLIENT_SCO_SHUTDOWN_ST: Ignoring event 3
|
||||
E (100427) BT_HF: APP HFP event: CONNECTION_STATE_EVT
|
||||
E (100437) BT_HF: --connection state disconnected, peer feats 0x0, chld_feats 0x0
|
||||
```
|
||||
|
||||
**Note: Only after HFP service are initiated and a service level connection exists between the HF Unit and AG device that other commands are available.**
|
||||
|
||||
And you can type `hf dis;` to disconnect with the connected AG device, and log prints like:
|
||||
|
||||
```
|
||||
E (100147) CNSL: Command [hf dis;]
|
||||
disconnect
|
||||
W (100427) BT_RFCOMM: port_rfc_closed RFCOMM connection in state 3 closed: Closed (res: 19)
|
||||
W (100427) BT_APPL: BTA_HF_CLIENT_SCO_SHUTDOWN_ST: Ignoring event 3
|
||||
E (100427) BT_HF: APP HFP event: CONNECTION_STATE_EVT
|
||||
E (100437) BT_HF: --connection state disconnected, peer feats 0x0, chld_feats 0x0
|
||||
W (93702) BT_RFCOMM: port_rfc_closed RFCOMM connection in state 3 closed: Closed (res: 19)
|
||||
W (93712) BT_APPL: BTA_HF_CLIENT_SCO_SHUTDOWN_ST: Ignoring event 3
|
||||
I (93712) BT_HF: APP HFP event: CONNECTION_STATE_EVT
|
||||
I (93712) BT_HF: --connection state disconnected, peer feats 0x0, chld_feats 0x0
|
||||
```
|
||||
|
||||
### Audio Connection and Disconnection
|
||||
|
||||
You can type `hf cona;` to establish the audio connection between HF Unit and AG device. Also, you can type `hf disa;` to close the audio data stream.
|
||||
You can type `hf cona;` to establish the audio connection between HF Unit and AG device. Log prints such as:
|
||||
|
||||
#### Situations for Audio Connection
|
||||
```
|
||||
E (117232) CNSL: Command [hf cona;]
|
||||
connect audio
|
||||
I (117232) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
I (117232) BT_HF: --audio state connecting
|
||||
E (117262) BT_BTM: btm_sco_connected, handle 181
|
||||
I (117262) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
I (117262) BT_HF: --audio state connected
|
||||
```
|
||||
|
||||
Also, you can type `hf disa;` to close the audio data stream. Log prints such as:
|
||||
|
||||
```
|
||||
E (133002) CNSL: Command [hf disa;]
|
||||
disconnect audio
|
||||
I (133262) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
I (133262) BT_HF: --audio state disconnected
|
||||
```
|
||||
#### Scenarios for Audio Connection
|
||||
|
||||
- Answer an incoming call
|
||||
- Enable voice recognition
|
||||
- Dial a outgoing call
|
||||
- Dial an outgoing call
|
||||
|
||||
#### Situations for Audio Disconnection
|
||||
#### Scenarios for Audio Disconnection
|
||||
|
||||
- Reject an incoming call
|
||||
- Disable the voice recognition
|
||||
|
||||
#### Audio Data path
|
||||
#### Choise of Codec
|
||||
|
||||
ESP32 supports two type of audio data pth: PCM and HCI.
|
||||
ESP32 supports both CVSD and mSBC codec. HF Unit and AG device determine which codec to use by exchanging features during service level connection. The choice of codec also depends on the your configuration in `menuconfig`.
|
||||
|
||||
- PCM : When using PCM audio data stream is "matrixed" to GPIO pins and you should link these GPIO pins to a speaker via i2s port.
|
||||
- HCI : When using HCI audio data stream will act in "loopback" mode. For example, you can hear your own voice when you place a call to a phone connected with a ESP32 development borad.
|
||||
|
||||
#### Codec
|
||||
|
||||
ESP32 support both CVSD and mSBC codec. HF Unit and AG devices determine which encoding to use when establishing a service level connection. The choice of codec method also depends on the user's configuration options for this example. Since CVSD is the default codec method in HFP, the situation using mSBC is showed here:
|
||||
Since CVSD is the default codec in HFP, we just show the scenarios using mSBC:
|
||||
|
||||
- If you enable `BT_HFP_WBS_ENABLE` in `menuconfig`, mSBC will be available.
|
||||
- If both HF Unit and AG support mSBC and `BT_HFP_WBS_ENABLE` is enabled, ESP32 chooses mSBC.
|
||||
@ -132,107 +169,98 @@ ESP32 support both CVSD and mSBC codec. HF Unit and AG devices determine which e
|
||||
|
||||
#### Answer an incoming call
|
||||
|
||||
You can type `hf ac;` to answer an incoming call and log prints like:
|
||||
You can type `hf ac;` to answer an incoming call and log prints such as:
|
||||
|
||||
```
|
||||
E (11278) CNSL: Command [hf ac;]
|
||||
E (196982) CNSL: Command [hf ac;]
|
||||
Answer call
|
||||
E (754946) BT_HF: APP HFP event: AT_RESPONSE
|
||||
E (754946) BT_HF: --AT response event, code 0, cme 0
|
||||
E (755326) BT_HF: APP HFP event: CALL_IND_EVT
|
||||
E (755326) BT_HF: --Call indicator call in progress
|
||||
E (755336) BT_HF: APP HFP event: CALL_SETUP_IND_EVT
|
||||
E (755336) BT_HF: --Call setup indicator NONE
|
||||
E (755966) BT_HF: APP HFP event: SIGNAL_STRENGTH_IND_EVT
|
||||
E (755966) BT_HF: -- signal strength: 5 (option)
|
||||
E (756206) BT_APPL: bta_dm_pm_btm_status hci_status=32
|
||||
I (197102) BT_HF: APP HFP event: AT_RESPONSE
|
||||
I (197102) BT_HF: --AT response event, code 0, cme 0
|
||||
E (197232) BT_BTM: btm_sco_connected, handle 181
|
||||
I (197232) BT_HF: APP HFP event: CALL_IND_EVT
|
||||
I (197232) BT_HF: --Call indicator call in progress
|
||||
I (197232) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
I (197242) BT_HF: --audio state connected
|
||||
```
|
||||
|
||||
#### Reject an incoming call
|
||||
|
||||
You can type `hf rc;` to reject an incoming call and log prints like:
|
||||
You can type `hf rc;` to reject an incoming call and log prints such as:
|
||||
|
||||
```
|
||||
E (717326) CNSL: Command [hf rc;]
|
||||
E (210112) CNSL: Command [hf rc;]
|
||||
Reject call
|
||||
E (717356) BT_HF: APP HFP event: AT_RESPONSE
|
||||
E (717356) BT_HF: --AT response event, code 0, cme 0
|
||||
E (717666) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
E (717666) BT_HF: --audio state disconnected
|
||||
E (717676) BT_HF: APP HFP event: CALL_SETUP_IND_EVT
|
||||
E (717676) BT_HF: --Call setup indicator NONE
|
||||
I (210822) BT_HF: APP HFP event: AT_RESPONSE
|
||||
I (210822) BT_HF: --AT response event, code 0, cme 0
|
||||
I (210842) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
I (210842) BT_HF: --audio state disconnected
|
||||
I (210902) BT_HF: APP HFP event: CALL_IND_EVT
|
||||
I (210902) BT_HF: --Call indicator NO call in progress
|
||||
```
|
||||
|
||||
#### Dial Number
|
||||
|
||||
This example supports three dialing commands, you can choose one from them:
|
||||
This example supports three dialing commands:
|
||||
|
||||
- `hf d <num>;` Dial the specific number.
|
||||
- `hf rd;` Redial the last number.
|
||||
- `hf dm <index>` Dial the specific indexed number in the AG memory.
|
||||
|
||||
When you type `hf d 186xxxx5549;` your phone will put an outgoing call to 186xxxx5549 and log prints like:
|
||||
For example, type `hf d 186xxxx5549;` to make an outgoing call to `186xxxx5549` and log prints such as:
|
||||
|
||||
```
|
||||
E (176927) CNSL: Command [hf d 186xxxx5549;]
|
||||
E (228882) CNSL: Command [hf d 186xxxx5549;]
|
||||
Dial number 186xxxx5549
|
||||
E (15246) BT_HF: APP HFP event: AT_RESPONSE
|
||||
E (15246) BT_HF: --AT response event, code 0, cme 0
|
||||
E (15246) BT_HF: APP HFP event: CALL_SETUP_IND_EVT
|
||||
E (15246) BT_HF: --Call setup indicator OUTGOING_DIALING
|
||||
E (15736) BT_HF: APP HFP event: CALL_SETUP_IND_EVT
|
||||
E (15736) BT_HF: --Call setup indicator OUTGOING_ALERTING
|
||||
E (15816) BT_BTM: btm_sco_connected, handle 180
|
||||
E (15816) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
E (15816) BT_HF: --audio state connected_msbc
|
||||
E (229702) BT_BTM: btm_sco_connected, handle 181
|
||||
I (229712) BT_HF: APP HFP event: CALL_SETUP_IND_EVT
|
||||
I (229712) BT_HF: --Call setup indicator OUTGOING_DIALING
|
||||
I (229712) BT_HF: APP HFP event: CALL_IND_EVT
|
||||
I (229712) BT_HF: --Call indicator call in progress
|
||||
I (229722) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
I (229722) BT_HF: --audio state connected
|
||||
I (229732) BT_HF: APP HFP event: CALL_SETUP_IND_EVT
|
||||
I (229732) BT_HF: --Call setup indicator NONE
|
||||
```
|
||||
|
||||
#### Respond and Hold
|
||||
|
||||
You can type `hf rh <btrh>;` to respond or hold the current call. The parameter set:
|
||||
You can type `hf rh <btrh>;` to respond or hold the current call. The parameter should be set as follows:
|
||||
|
||||
- `<btrh>` : 0 - hold current call, 1 - answer held call, 2 - end held call.
|
||||
|
||||
#### Volume Control
|
||||
|
||||
You can type `hf vu <tgt> <vol>;` to sync volume gain of headset or microphone. The parameter set:
|
||||
You can type `hf vu <tgt> <vol>;` to update volume gain of speaker or microphone. The parameter should be set as follows:
|
||||
|
||||
- `<tgt>` : 0 - headset, 1 - microphone.
|
||||
- `<tgt>` : 0 - speaker, 1 - microphone.
|
||||
- `<vol>` : Integer among 0 - 15.
|
||||
|
||||
For example, `hf vu 0 9;` sync the volume of headset and log prints like:
|
||||
For example, type `hf vu 0 9;` to update the volume of speaker and log on AG prints:
|
||||
|
||||
```
|
||||
E (34827) CNSL: Command [hf vu 0 9;]
|
||||
volume update
|
||||
E (35467) BT_HF: APP HFP event: AT_RESPONSE
|
||||
E (35467) BT_HF: --AT response event, code 0, cme 0
|
||||
|
||||
I (43684) BT_APP_HF: APP HFP event: VOLUME_CONTROL_EVT
|
||||
I (43684) BT_APP_HF: --Volume Target: SPEAKER, Volume 9
|
||||
```
|
||||
|
||||
And also, `hf vu 1 9;` sync the volume gain of microphone and log prints like:
|
||||
And also, `hf vu 1 9;` update the volume gain of microphone and log on AG prints:
|
||||
|
||||
```
|
||||
E (39247) CNSL: Command [hf vu 1 9;]
|
||||
volume updateE (39257)
|
||||
BT_HF: APP HFP event: AT_RESPONSEE (39257)
|
||||
BT_HF: --AT response event, code 0, cme 0
|
||||
|
||||
I (177254) BT_APP_HF: APP HFP event: VOLUME_CONTROL_EVT
|
||||
I (177254) BT_APP_HF: --Volume Target: MICROPHONE, Volume 9
|
||||
```
|
||||
|
||||
#### Voice Recognition
|
||||
|
||||
You can type `hf vron;` to start the voice recognition of AG and type `hf vroff;` to end the voice recognition. For example, type `hf vron;` and log prints like:
|
||||
You can type `hf vron;` to start the voice recognition of AG and type `hf vroff;` to terminate this function. For example, type `hf vron;` and log prints such as:
|
||||
|
||||
```
|
||||
E (45002) CNSL: Command [hf vron;]
|
||||
E (292432) CNSL: Command [hf vron;]
|
||||
Start voice recognition
|
||||
E (25162) BT_HF: APP HFP event: AT_RESPONSE
|
||||
E (25162) BT_HF: --AT response event, code 0, cme 0
|
||||
E (25182) BT_HF: APP HFP event: VOLUME_CONTROL_EVT
|
||||
E (25182) BT_HF: --volume_target: SPEAKER, volume 7
|
||||
E (191202) BT_HF: --VR state enabled
|
||||
|
||||
I (293172) BT_HF: APP HFP event: AT_RESPONSE
|
||||
I (293172) BT_HF: --AT response event, code 0, cme 0
|
||||
E (293702) BT_BTM: btm_sco_connected, handle 181
|
||||
I (293702) BT_HF: APP HFP event: AUDIO_STATE_EVT
|
||||
I (293702) BT_HF: --audio state connecte
|
||||
```
|
||||
|
||||
#### Query Current Operator Name
|
||||
@ -240,27 +268,25 @@ E (191202) BT_HF: --VR state enabled
|
||||
You can type `hf qop;` to query the current operator name and log prints like:
|
||||
|
||||
```
|
||||
E (45002) CNSL: Command [hf qop;]
|
||||
E (338322) CNSL: Command [hf qop;]
|
||||
Query operator
|
||||
E (393672) BT_HF: APP HFP event: CURRENT_OPERATOR_EVT
|
||||
E (393672) BT_HF: --operator name: 中国联通
|
||||
E (393672) BT_HF: APP HFP event: AT_RESPONSE
|
||||
E (393682) BT_HF: --AT response event, code 0, cme 0
|
||||
|
||||
I (339202) BT_HF: APP HFP event: CURRENT_OPERATOR_EVT
|
||||
I (339202) BT_HF: --operator name: 中国联通
|
||||
I (339202) BT_HF: APP HFP event: AT_RESPONSE
|
||||
I (339202) BT_HF: --AT response event, code 0, cme 0
|
||||
```
|
||||
|
||||
#### Retrieve Subscriber Information
|
||||
|
||||
You can type `hf rs;` to retrieve subscriber information and log prints like:
|
||||
You can type `hf rs;` to retrieve subscriber information and log prints such as:
|
||||
|
||||
```
|
||||
E (231427) CNSL: Command [hf rs;]
|
||||
E (352902) CNSL: Command [hf rs;]
|
||||
Retrieve subscriber information
|
||||
E (231447) BT_HF: APP HFP event: SUBSCRIBER_INFO_EVT
|
||||
E (231447) BT_HF: --subscriber type voice, number +86156xxxx1969
|
||||
E (231447) BT_HF: APP HFP event: AT_RESPONSE
|
||||
E (231447) BT_HF: --AT response event, code 0, cme
|
||||
|
||||
I (353702) BT_HF: APP HFP event: SUBSCRIBER_INFO_EVT
|
||||
I (353702) BT_HF: --subscriber type unknown, number 186xxxx5549
|
||||
I (353702) BT_HF: APP HFP event: AT_RESPONSE
|
||||
I (353702) BT_HF: --AT response event, code 0, cme 0
|
||||
```
|
||||
|
||||
#### Query Current Call Status
|
||||
@ -268,35 +294,39 @@ E (231447) BT_HF: --AT response event, code 0, cme
|
||||
You can type `hf qc;` to query current call status and log prints like:
|
||||
|
||||
```
|
||||
E (102837) CNSL: Command [hf qc;]
|
||||
E (354522) CNSL: Command [hf qc;]
|
||||
Query current call status
|
||||
E (29228) BT_HF: APP HFP event: CLCC_EVT
|
||||
E (29228) BT_HF: --Current call: idx 1, dir incoming, state active, mpty single, number 186xxxx5549
|
||||
E (29238) BT_HF: APP HFP event: AT_RESPONSE
|
||||
E (29238) BT_HF: --AT response event, code 0, cme 0
|
||||
I (354582) BT_HF: APP HFP event: CLCC_EVT
|
||||
I (354582) BT_HF: --Current call: idx 1, dir incoming, state active, mpty single, number 186xxxx5549
|
||||
I (354582) BT_HF: APP HFP event: AT_RESPONSE
|
||||
I (354592) BT_HF: --AT response event, code 0, cme 0
|
||||
```
|
||||
|
||||
#### Transport DTMF Code
|
||||
|
||||
You can type `hf k <dtmf>;` to transport a DTMF code to AG. Log on HF unit side prints like:`send dtmf code: 9` and log on AG side prints such as:
|
||||
|
||||
```
|
||||
I (196284) BT_APP_HF: APP HFP event: DTMF_RESPONSE_EVT
|
||||
I (196284) BT_APP_HF: --DTMF code is: 9.
|
||||
```
|
||||
|
||||
## Troubleshooting
|
||||
|
||||
- You should type the command in the terminal according to the format described in the command help table.
|
||||
- Not all commands in the table are supported by AG device.
|
||||
- If you want to use HF Unit to establish a service level connection with AG, you should add the MAC address of AG in `hfp_hf/main/bt_app.c`, for example:
|
||||
If you encounter any problems, please check if the following rules are followed:
|
||||
|
||||
```
|
||||
esp_bd_addr_t peer_addr = {0xb4, 0xe6, 0x2d, 0xeb, 0x09, 0x93};
|
||||
|
||||
```
|
||||
|
||||
- Use `esp_hf_client_register_callback()` and `esp_hf_client_init();` before establishing a service level connection.
|
||||
- You should type the command in the terminal according to the format described in the commands help table.
|
||||
- Not all commands in the table are supported by AG device like _Hands Free Audio Gateway (hfp_ag)_ example from ESP-IDF.
|
||||
- If you want to use `hf con;` to establish a service level connection with specific AG device, you should add the MAC address of the AG device in `bt_app.c`, for example: `esp_bd_addr_t peer_addr = {0xb4, 0xe6, 0x2d, 0xeb, 0x09, 0x93};`
|
||||
- Use `esp_hf_client_register_callback()` and `esp_hf_client_init();` before establishing a service level connection.
|
||||
|
||||
## Example Breakdown
|
||||
|
||||
This example has relatively more source files than other bluetooth examples because _Hands Free Profile_ is somehow complex. But we want to show the functions of _Hands Free Profile_ in a simple way, so we use the _Commands and Effects_ scheme to show the usage of APIs of HFP in ESP-IDF.
|
||||
Due to the complexity of HFP, this example has more source files than other bluetooth examples. To show functions of HFP in a simple way, we use the Commands and Effects scheme to illustrate APIs of HFP in ESP-IDF.
|
||||
|
||||
- The example will respond to user command through UART console. Please go to `hfp_hf/main/console_uart.c` for the configuration details.
|
||||
- For voice interface, ESP32 has provided PCM input/output signals which can be **matrixed** to GPIOs, please go to `hfp_hf/main/gpio_pcm_config.c` for the configuration details.
|
||||
- If you want to fix the command table, please refer to `hfp_hf/main/app_hf_msg_set.c`.
|
||||
- If you want to fix the command parse rules, please refer to `hfp_hf/main/app_hf_msg_prs.c`.
|
||||
- If you want to fix the responses of HF Unit or want to fix the log, please refer to `hfp_hf/main/bt_app_hf.c`.
|
||||
- Task configuration part is in `hfp_hf/main/bt_app_core.c`.
|
||||
- The example will respond to user command through UART console. Please go to `console_uart.c` for the configuration details.
|
||||
- For voice interface, ESP32 has provided PCM input/output signals which can be mapped to GPIO pins. So, please go to `gpio_pcm_config.c` for the configuration details.
|
||||
- If you want to update the command table, please refer to `app_hf_msg_set.c`.
|
||||
- If you want to update the command parse rules, please refer to `app_hf_msg_prs.c`.
|
||||
- If you want to update the responses of HF Unit or want to update the log, please refer to `bt_app_hf.c`.
|
||||
- Task configuration part is in `bt_app_core.c`.
|
@ -17,7 +17,7 @@ void hf_msg_show_usage(void)
|
||||
{
|
||||
printf("########################################################################\n");
|
||||
printf("HF client command usage manual\n");
|
||||
printf("HF client commands begin with \"hf\" and ends with \";\"\n");
|
||||
printf("HF client commands begin with \"hf\" and end with \";\"\n");
|
||||
printf("Supported commands are as follows, arguments are embraced with < and >\n");
|
||||
printf("hf con; -- setup connection with peer device\n");
|
||||
printf("hf dis; -- release connection with peer device\n");
|
||||
@ -27,22 +27,25 @@ void hf_msg_show_usage(void)
|
||||
printf("hf qc; -- query current call status\n");
|
||||
printf("hf ac; -- answer incoming call\n");
|
||||
printf("hf rc; -- reject incoming call\n");
|
||||
printf("hf d <num>; -- dial <num>, e.g. hf d 11223344\n");
|
||||
printf("hf rd; -- redial\n");
|
||||
printf("hf dm <index>; -- dial memory\n");
|
||||
printf("hf vron; -- start voice recognition\n");
|
||||
printf("hf vroff; -- stop voice recognition\n");
|
||||
printf("hf vu <tgt> <vol>; -- volume update, tgt: 0-speaker, 1-microphone\n");
|
||||
printf("hf d <num>; -- dial <num>, e.g. hf d 11223344\n");
|
||||
printf("hf rd; -- redial\n");
|
||||
printf("hf dm <index>; -- dial memory\n");
|
||||
printf("hf vron; -- start voice recognition\n");
|
||||
printf("hf vroff; -- stop voice recognition\n");
|
||||
printf("hf vu <tgt> <vol>; -- volume update\n");
|
||||
printf(" tgt: 0-speaker, 1-microphone\n");
|
||||
printf(" vol: volume gain ranges from 0 to 15\n");
|
||||
printf("hf rs; -- retrieve subscriber information\n");
|
||||
printf("hf rv; -- retrieve last voice tag number\n");
|
||||
printf("hf rh <btrh>; -- response and hold, btrh: 0 - put call on hold,\n");
|
||||
printf(" 1 - accept the held call, 2 -reject the held call\n");
|
||||
printf("hf k <dtmf>; -- send dtmf code. dtmf: single character in set \n");
|
||||
printf(" 0-9, *, #, A-D\n");
|
||||
printf("hf h; -- show command manual\n");
|
||||
printf("hf rs; -- retrieve subscriber information\n");
|
||||
printf("hf rv; -- retrieve last voice tag number\n");
|
||||
printf("hf rh <btrh>; -- response and hold\n");
|
||||
printf(" btrh:\n");
|
||||
printf(" 0 - put call on hold,\n");
|
||||
printf(" 1 - accept the held call,\n");
|
||||
printf(" 2 -reject the held call\n");
|
||||
printf("hf k <dtmf>; -- send dtmf code.\n");
|
||||
printf(" dtmf: single character in set 0-9, *, #, A-D\n");
|
||||
printf("hf h; -- show command manual\n");
|
||||
printf("########################################################################\n");
|
||||
|
||||
}
|
||||
|
||||
#define HF_CMD_HANDLER(cmd) static void hf_##cmd##_handler(int argn, char **argv)
|
||||
|
@ -3,7 +3,7 @@
|
||||
|
||||
Unless required by applicable law or agreed to in writing, this
|
||||
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
||||
CONDITIONS OF ANY KIND, either express or implied.
|
||||
CONDITIONS OF ANY KIND, either express esp_hf_ag_apiied.
|
||||
*/
|
||||
|
||||
#include <stdint.h>
|
||||
@ -18,7 +18,6 @@
|
||||
#include "esp_bt_device.h"
|
||||
#include "esp_gap_bt_api.h"
|
||||
#include "esp_hf_client_api.h"
|
||||
|
||||
#include "freertos/FreeRTOS.h"
|
||||
#include "freertos/task.h"
|
||||
#include "freertos/queue.h"
|
||||
@ -169,9 +168,9 @@ const char *c_inband_ring_state_str[] = {
|
||||
"Provided",
|
||||
};
|
||||
|
||||
esp_bd_addr_t peer_addr;
|
||||
// If you want to use a specific MAC address just define peer_address here
|
||||
// esp_bd_addr_t peer_addr = {0xb4, 0xe6, 0x2d, 0xeb, 0x09, 0x93};
|
||||
// esp_bd_addr_t peer_addr;
|
||||
// If you want to connect a specific device, add it's address here
|
||||
esp_bd_addr_t peer_addr = {0xb4, 0xe6, 0x2d, 0xeb, 0x09, 0x93};
|
||||
|
||||
#if CONFIG_BTDM_CONTROLLER_BR_EDR_SCO_DATA_PATH_HCI
|
||||
|
||||
@ -180,10 +179,6 @@ static RingbufHandle_t m_rb = NULL;
|
||||
|
||||
static void bt_app_hf_client_audio_open(void)
|
||||
{
|
||||
if (!m_rb) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
m_rb = xRingbufferCreate(ESP_HFP_RINGBUF_SIZE, RINGBUF_TYPE_BYTEBUF);
|
||||
}
|
||||
|
||||
@ -224,7 +219,7 @@ static void bt_app_hf_client_incoming_cb(const uint8_t *buf, uint32_t sz)
|
||||
}
|
||||
BaseType_t done = xRingbufferSend(m_rb, (uint8_t *)buf, sz, 0);
|
||||
if (! done) {
|
||||
ESP_LOGI(BT_HF_TAG, "rb send fail");
|
||||
ESP_LOGE(BT_HF_TAG, "rb send fail");
|
||||
}
|
||||
|
||||
esp_hf_client_outgoing_data_ready();
|
||||
@ -235,7 +230,7 @@ static void bt_app_hf_client_incoming_cb(const uint8_t *buf, uint32_t sz)
|
||||
void bt_app_hf_client_cb(esp_hf_client_cb_event_t event, esp_hf_client_cb_param_t *param)
|
||||
{
|
||||
if (event <= ESP_HF_CLIENT_RING_IND_EVT) {
|
||||
ESP_LOGE(BT_HF_TAG, "APP HFP event: %s", c_hf_evt_str[event]);
|
||||
ESP_LOGI(BT_HF_TAG, "APP HFP event: %s", c_hf_evt_str[event]);
|
||||
} else {
|
||||
ESP_LOGE(BT_HF_TAG, "APP HFP invalid event %d", event);
|
||||
}
|
||||
@ -243,7 +238,7 @@ void bt_app_hf_client_cb(esp_hf_client_cb_event_t event, esp_hf_client_cb_param_
|
||||
switch (event) {
|
||||
case ESP_HF_CLIENT_CONNECTION_STATE_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--connection state %s, peer feats 0x%x, chld_feats 0x%x",
|
||||
ESP_LOGI(BT_HF_TAG, "--connection state %s, peer feats 0x%x, chld_feats 0x%x",
|
||||
c_connection_state_str[param->conn_stat.state],
|
||||
param->conn_stat.peer_feat,
|
||||
param->conn_stat.chld_feat);
|
||||
@ -253,7 +248,7 @@ void bt_app_hf_client_cb(esp_hf_client_cb_event_t event, esp_hf_client_cb_param_
|
||||
|
||||
case ESP_HF_CLIENT_AUDIO_STATE_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--audio state %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--audio state %s",
|
||||
c_audio_state_str[param->audio_stat.state]);
|
||||
#if CONFIG_BTDM_CONTROLLER_BR_EDR_SCO_DATA_PATH_HCI
|
||||
if (param->audio_stat.state == ESP_HF_CLIENT_AUDIO_STATE_CONNECTED ||
|
||||
@ -270,91 +265,91 @@ void bt_app_hf_client_cb(esp_hf_client_cb_event_t event, esp_hf_client_cb_param_
|
||||
|
||||
case ESP_HF_CLIENT_BVRA_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--VR state %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--VR state %s",
|
||||
c_vr_state_str[param->bvra.value]);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CIND_SERVICE_AVAILABILITY_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--NETWORK STATE %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--NETWORK STATE %s",
|
||||
c_service_availability_status_str[param->service_availability.status]);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CIND_ROAMING_STATUS_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--ROAMING: %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--ROAMING: %s",
|
||||
c_roaming_status_str[param->roaming.status]);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CIND_SIGNAL_STRENGTH_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "-- signal strength: %d",
|
||||
ESP_LOGI(BT_HF_TAG, "-- signal strength: %d",
|
||||
param->signal_strength.value);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CIND_BATTERY_LEVEL_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--battery level %d",
|
||||
ESP_LOGI(BT_HF_TAG, "--battery level %d",
|
||||
param->battery_level.value);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_COPS_CURRENT_OPERATOR_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--operator name: %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--operator name: %s",
|
||||
param->cops.name);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CIND_CALL_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--Call indicator %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--Call indicator %s",
|
||||
c_call_str[param->call.status]);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CIND_CALL_SETUP_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--Call setup indicator %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--Call setup indicator %s",
|
||||
c_call_setup_str[param->call_setup.status]);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CIND_CALL_HELD_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--Call held indicator %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--Call held indicator %s",
|
||||
c_call_held_str[param->call_held.status]);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_BTRH_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--response and hold %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--response and hold %s",
|
||||
c_resp_and_hold_str[param->btrh.status]);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CLIP_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--clip number %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--clip number %s",
|
||||
(param->clip.number == NULL) ? "NULL" : (param->clip.number));
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CCWA_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--call_waiting %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--call_waiting %s",
|
||||
(param->ccwa.number == NULL) ? "NULL" : (param->ccwa.number));
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CLCC_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--Current call: idx %d, dir %s, state %s, mpty %s, number %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--Current call: idx %d, dir %s, state %s, mpty %s, number %s",
|
||||
param->clcc.idx,
|
||||
c_call_dir_str[param->clcc.dir],
|
||||
c_call_state_str[param->clcc.status],
|
||||
@ -365,7 +360,7 @@ void bt_app_hf_client_cb(esp_hf_client_cb_event_t event, esp_hf_client_cb_param_
|
||||
|
||||
case ESP_HF_CLIENT_VOLUME_CONTROL_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--volume_target: %s, volume %d",
|
||||
ESP_LOGI(BT_HF_TAG, "--volume_target: %s, volume %d",
|
||||
c_volume_control_target_str[param->volume_control.type],
|
||||
param->volume_control.volume);
|
||||
break;
|
||||
@ -373,14 +368,14 @@ void bt_app_hf_client_cb(esp_hf_client_cb_event_t event, esp_hf_client_cb_param_
|
||||
|
||||
case ESP_HF_CLIENT_AT_RESPONSE_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--AT response event, code %d, cme %d",
|
||||
ESP_LOGI(BT_HF_TAG, "--AT response event, code %d, cme %d",
|
||||
param->at_response.code, param->at_response.cme);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_CNUM_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--subscriber type %s, number %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--subscriber type %s, number %s",
|
||||
c_subscriber_service_type_str[param->cnum.type],
|
||||
(param->cnum.number == NULL) ? "NULL" : param->cnum.number);
|
||||
break;
|
||||
@ -388,14 +383,14 @@ void bt_app_hf_client_cb(esp_hf_client_cb_event_t event, esp_hf_client_cb_param_
|
||||
|
||||
case ESP_HF_CLIENT_BSIR_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--inband ring state %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--inband ring state %s",
|
||||
c_inband_ring_state_str[param->bsir.state]);
|
||||
break;
|
||||
}
|
||||
|
||||
case ESP_HF_CLIENT_BINP_EVT:
|
||||
{
|
||||
ESP_LOGE(BT_HF_TAG, "--last voice tag number: %s",
|
||||
ESP_LOGI(BT_HF_TAG, "--last voice tag number: %s",
|
||||
(param->binp.number == NULL) ? "NULL" : param->binp.number);
|
||||
break;
|
||||
}
|
||||
|
@ -64,31 +64,31 @@ static void console_uart_task(void *pvParameters)
|
||||
//Event of HW FIFO overflow detected
|
||||
case UART_FIFO_OVF:
|
||||
{
|
||||
ESP_LOGI(TAG_CNSL, "hw fifo overflow\n");
|
||||
ESP_LOGI(TAG_CNSL, "hw fifo overflow");
|
||||
break;
|
||||
}
|
||||
//Event of UART ring buffer full
|
||||
case UART_BUFFER_FULL:
|
||||
{
|
||||
ESP_LOGI(TAG_CNSL, "ring buffer full\n");
|
||||
ESP_LOGI(TAG_CNSL, "ring buffer full");
|
||||
break;
|
||||
}
|
||||
//Event of UART RX break detected
|
||||
case UART_BREAK:
|
||||
{
|
||||
ESP_LOGI(TAG_CNSL, "uart rx break\n");
|
||||
ESP_LOGI(TAG_CNSL, "uart rx break");
|
||||
break;
|
||||
}
|
||||
//Event of UART parity check error
|
||||
case UART_PARITY_ERR:
|
||||
{
|
||||
ESP_LOGI(TAG_CNSL, "uart parity error\n");
|
||||
ESP_LOGI(TAG_CNSL, "uart parity error");
|
||||
break;
|
||||
}
|
||||
//Event of UART frame error
|
||||
case UART_FRAME_ERR:
|
||||
{
|
||||
ESP_LOGI(TAG_CNSL, "uart frame error\n");
|
||||
ESP_LOGI(TAG_CNSL, "uart frame error");
|
||||
break;
|
||||
}
|
||||
//Others
|
||||
@ -107,7 +107,7 @@ esp_err_t console_uart_init(void)
|
||||
|
||||
ret = uart_param_config(CONSOLE_UART_NUM, &uart_cfg);
|
||||
if (ret != ESP_OK) {
|
||||
ESP_LOGE(TAG_CNSL, "Uart %d initialize err %04x\n", CONSOLE_UART_NUM, ret);
|
||||
ESP_LOGE(TAG_CNSL, "Uart %d initialize err %04x", CONSOLE_UART_NUM, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user