xref: /btstack/test/hfp/mock.c (revision b51c851cc9d1d214734076a917fb20e3bcb1d8cc)
1dc5611fdSMilanka Ringwald /*
2dc5611fdSMilanka Ringwald  * Copyright (C) 2014 BlueKitchen GmbH
3dc5611fdSMilanka Ringwald  *
4dc5611fdSMilanka Ringwald  * Redistribution and use in source and binary forms, with or without
5dc5611fdSMilanka Ringwald  * modification, are permitted provided that the following conditions
6dc5611fdSMilanka Ringwald  * are met:
7dc5611fdSMilanka Ringwald  *
8dc5611fdSMilanka Ringwald  * 1. Redistributions of source code must retain the above copyright
9dc5611fdSMilanka Ringwald  *    notice, this list of conditions and the following disclaimer.
10dc5611fdSMilanka Ringwald  * 2. Redistributions in binary form must reproduce the above copyright
11dc5611fdSMilanka Ringwald  *    notice, this list of conditions and the following disclaimer in the
12dc5611fdSMilanka Ringwald  *    documentation and/or other materials provided with the distribution.
13dc5611fdSMilanka Ringwald  * 3. Neither the name of the copyright holders nor the names of
14dc5611fdSMilanka Ringwald  *    contributors may be used to endorse or promote products derived
15dc5611fdSMilanka Ringwald  *    from this software without specific prior written permission.
16dc5611fdSMilanka Ringwald  * 4. Any redistribution, use, or modification is done solely for
17dc5611fdSMilanka Ringwald  *    personal benefit and not for any commercial purpose or for
18dc5611fdSMilanka Ringwald  *    monetary gain.
19dc5611fdSMilanka Ringwald  *
20dc5611fdSMilanka Ringwald  * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
21dc5611fdSMilanka Ringwald  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22dc5611fdSMilanka Ringwald  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23dc5611fdSMilanka Ringwald  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS
24dc5611fdSMilanka Ringwald  * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25dc5611fdSMilanka Ringwald  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26dc5611fdSMilanka Ringwald  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
27dc5611fdSMilanka Ringwald  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
28dc5611fdSMilanka Ringwald  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
29dc5611fdSMilanka Ringwald  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
30dc5611fdSMilanka Ringwald  * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31dc5611fdSMilanka Ringwald  * SUCH DAMAGE.
32dc5611fdSMilanka Ringwald  *
33dc5611fdSMilanka Ringwald  * Please inquire about commercial licensing options at
34dc5611fdSMilanka Ringwald  * [email protected]
35dc5611fdSMilanka Ringwald  *
36dc5611fdSMilanka Ringwald  */
37dc5611fdSMilanka Ringwald 
38dc5611fdSMilanka Ringwald // *****************************************************************************
39dc5611fdSMilanka Ringwald //
40dc5611fdSMilanka Ringwald // HFP BTstack Mocks
41dc5611fdSMilanka Ringwald //
42dc5611fdSMilanka Ringwald // *****************************************************************************
43dc5611fdSMilanka Ringwald 
4412381011SMilanka Ringwald #include <stdint.h>
4512381011SMilanka Ringwald #include <stdio.h>
4612381011SMilanka Ringwald #include <stdlib.h>
4712381011SMilanka Ringwald #include <string.h>
4812381011SMilanka Ringwald 
4912381011SMilanka Ringwald #include <btstack/btstack.h>
5012381011SMilanka Ringwald #include "hci.h"
5112381011SMilanka Ringwald #include "hci_dump.h"
5212381011SMilanka Ringwald #include "sdp_query_rfcomm.h"
5312381011SMilanka Ringwald #include "rfcomm.h"
54dc5611fdSMilanka Ringwald #include "hfp_hf.h"
55dc5611fdSMilanka Ringwald 
56dc5611fdSMilanka Ringwald #include "mock.h"
5712381011SMilanka Ringwald 
5812381011SMilanka Ringwald static void *registered_sdp_app_context;
5912381011SMilanka Ringwald static uint8_t sdp_rfcomm_channel_nr = 1;
6012381011SMilanka Ringwald const char sdp_rfcomm_service_name[] = "BTstackMock";
6112381011SMilanka Ringwald static uint16_t rfcomm_cid = 1;
626f1de21cSMilanka Ringwald static bd_addr_t dev_addr;
636f1de21cSMilanka Ringwald static uint16_t sco_handle = 10;
6412381011SMilanka Ringwald static uint8_t rfcomm_payload[200];
6512381011SMilanka Ringwald static uint16_t rfcomm_payload_len;
6612381011SMilanka Ringwald void * active_connection;
678c72a38bSMilanka Ringwald hfp_connection_t * hfp_context;
6812381011SMilanka Ringwald 
6912381011SMilanka Ringwald void (*registered_rfcomm_packet_handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size);
7012381011SMilanka Ringwald void (*registered_sdp_app_callback)(sdp_query_event_t * event, void * context);
7112381011SMilanka Ringwald 
7212381011SMilanka Ringwald uint8_t * get_rfcomm_payload(){
7312381011SMilanka Ringwald 	return &rfcomm_payload[0];
7412381011SMilanka Ringwald }
7512381011SMilanka Ringwald 
7612381011SMilanka Ringwald uint16_t get_rfcomm_payload_len(){
7712381011SMilanka Ringwald 	return rfcomm_payload_len;
7812381011SMilanka Ringwald }
7912381011SMilanka Ringwald 
80fc6b77dcSMilanka Ringwald static void prepare_rfcomm_buffer(uint8_t * data, int len){
81656f7353SMilanka Ringwald     if (len <= 0) return;
8212381011SMilanka Ringwald     memset(&rfcomm_payload, 0, 200);
83fc6b77dcSMilanka Ringwald 	int pos = 0;
84656f7353SMilanka Ringwald 
85656f7353SMilanka Ringwald     if (strncmp((char*)data, "AT", 2) == 0){
86656f7353SMilanka Ringwald         strncpy((char*)&rfcomm_payload[pos], (char*)data, len);
87656f7353SMilanka Ringwald         pos += len;
88656f7353SMilanka Ringwald     } else {
89fc6b77dcSMilanka Ringwald     	rfcomm_payload[pos++] = '\r';
90fc6b77dcSMilanka Ringwald 		rfcomm_payload[pos++] = '\n';
91656f7353SMilanka Ringwald         strncpy((char*)&rfcomm_payload[pos], (char*)data, len);
92fc6b77dcSMilanka Ringwald         pos += len;
93656f7353SMilanka Ringwald 
94656f7353SMilanka Ringwald         if (memcmp((char*)data, "+BAC", 4) != 0 &&
95656f7353SMilanka Ringwald             memcmp((char*)data, "+BCS", 4) != 0){
96fc6b77dcSMilanka Ringwald             rfcomm_payload[pos++] = '\r';
97dc5611fdSMilanka Ringwald             rfcomm_payload[pos++] = '\n';
98dc5611fdSMilanka Ringwald             rfcomm_payload[pos++] = 'O';
99dc5611fdSMilanka Ringwald             rfcomm_payload[pos++] = 'K';
100fc6b77dcSMilanka Ringwald         }
101656f7353SMilanka Ringwald 
102656f7353SMilanka Ringwald     }
103dc5611fdSMilanka Ringwald 	rfcomm_payload[pos++] = '\r';
104656f7353SMilanka Ringwald     rfcomm_payload[pos++] = '\n';
105656f7353SMilanka Ringwald     rfcomm_payload[pos] = 0;
106fc6b77dcSMilanka Ringwald 	rfcomm_payload_len = pos;
107fc6b77dcSMilanka Ringwald }
108fc6b77dcSMilanka Ringwald 
10962656e33SMilanka Ringwald static void print_without_newlines(uint8_t *data, uint16_t len){
11062656e33SMilanka Ringwald     int found_newline = 0;
11162656e33SMilanka Ringwald     int found_item = 0;
112c26caa21SMilanka Ringwald 
11362656e33SMilanka Ringwald     for (int i=0; i<len; i++){
11462656e33SMilanka Ringwald         if (data[i] == '\r' || data[i] == '\n'){
11562656e33SMilanka Ringwald             if (!found_newline && found_item) printf("\n");
11662656e33SMilanka Ringwald             found_newline = 1;
11762656e33SMilanka Ringwald         } else {
11862656e33SMilanka Ringwald             printf("%c", data[i]);
11962656e33SMilanka Ringwald             found_newline = 0;
12062656e33SMilanka Ringwald             found_item = 1;
12162656e33SMilanka Ringwald         }
12262656e33SMilanka Ringwald     }
12362656e33SMilanka Ringwald     printf("\n");
12462656e33SMilanka Ringwald }
125fc6b77dcSMilanka Ringwald 
1260fcfc799SMilanka Ringwald extern "C" void l2cap_init(void){}
1270fcfc799SMilanka Ringwald 
1280fcfc799SMilanka Ringwald extern "C" void l2cap_register_packet_handler(void (*handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size)){
1290fcfc799SMilanka Ringwald }
1300fcfc799SMilanka Ringwald 
1310fcfc799SMilanka Ringwald 
13212381011SMilanka Ringwald int  rfcomm_send_internal(uint16_t rfcomm_cid, uint8_t *data, uint16_t len){
133656f7353SMilanka Ringwald 	if (strncmp((char*)data, "AT", 2) == 0){
13462656e33SMilanka Ringwald 		printf("Verify HF state machine response: ");
13562656e33SMilanka Ringwald         print_without_newlines(data,len);
136fc6b77dcSMilanka Ringwald 	} else {
13762656e33SMilanka Ringwald         printf("Verify AG state machine response: ");
13862656e33SMilanka Ringwald         print_without_newlines(data,len);
139fc6b77dcSMilanka Ringwald 	}
140656f7353SMilanka Ringwald 	strncpy((char*)&rfcomm_payload[0], (char*)data, len);
141656f7353SMilanka Ringwald     rfcomm_payload_len = len;
14212381011SMilanka Ringwald 	return 0;
14312381011SMilanka Ringwald }
14412381011SMilanka Ringwald 
1456f1de21cSMilanka Ringwald static void hci_event_sco_complete(){
1468c72a38bSMilanka Ringwald     uint8_t event[19];
1476f1de21cSMilanka Ringwald     uint8_t pos = 0;
1486f1de21cSMilanka Ringwald     event[pos++] = HCI_EVENT_SYNCHRONOUS_CONNECTION_COMPLETE;
1496f1de21cSMilanka Ringwald     event[pos++] = sizeof(event) - 2;
1506f1de21cSMilanka Ringwald 
1516f1de21cSMilanka Ringwald     event[pos++] = 0; //status
1526f1de21cSMilanka Ringwald     bt_store_16(event,  pos, sco_handle);   pos += 2; // sco handle
1536f1de21cSMilanka Ringwald     bt_flip_addr(&event[pos], dev_addr);    pos += 6;
1546f1de21cSMilanka Ringwald 
1556f1de21cSMilanka Ringwald     event[pos++] = 0; // link_type
1566f1de21cSMilanka Ringwald     event[pos++] = 0; // transmission_interval
1576f1de21cSMilanka Ringwald     event[pos++] = 0; // retransmission_interval
1586f1de21cSMilanka Ringwald 
1596f1de21cSMilanka Ringwald     bt_store_16(event,  pos, 0);   pos += 2; // rx_packet_length
1606f1de21cSMilanka Ringwald     bt_store_16(event,  pos, 0);   pos += 2; // tx_packet_length
1616f1de21cSMilanka Ringwald 
1626f1de21cSMilanka Ringwald     event[pos++] = 0; // air_mode
1638c72a38bSMilanka Ringwald     (*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, event, sizeof(event));
1646f1de21cSMilanka Ringwald }
1656f1de21cSMilanka Ringwald 
16612381011SMilanka Ringwald int hci_send_cmd(const hci_cmd_t *cmd, ...){
1678c72a38bSMilanka Ringwald 	printf("hci_send_cmd opcode 0x%02x\n", cmd->opcode);
1686f1de21cSMilanka Ringwald     if (cmd->opcode == 0x428){
1696f1de21cSMilanka Ringwald         hci_event_sco_complete();
1706f1de21cSMilanka Ringwald     }
17112381011SMilanka Ringwald 	return 0;
17212381011SMilanka Ringwald }
17312381011SMilanka Ringwald 
17412381011SMilanka Ringwald 
17512381011SMilanka Ringwald void sdp_query_rfcomm_register_callback(void(*sdp_app_callback)(sdp_query_event_t * event, void * context), void * context){
17612381011SMilanka Ringwald 	registered_sdp_app_callback = sdp_app_callback;
17712381011SMilanka Ringwald 	registered_sdp_app_context = context;
17812381011SMilanka Ringwald }
17912381011SMilanka Ringwald 
18012381011SMilanka Ringwald static void sdp_query_complete_response(uint8_t status){
18112381011SMilanka Ringwald     sdp_query_complete_event_t complete_event = {
18212381011SMilanka Ringwald         SDP_QUERY_COMPLETE,
18312381011SMilanka Ringwald         status
18412381011SMilanka Ringwald     };
18512381011SMilanka Ringwald     (*registered_sdp_app_callback)((sdp_query_event_t*)&complete_event, registered_sdp_app_context);
18612381011SMilanka Ringwald }
18712381011SMilanka Ringwald 
18812381011SMilanka Ringwald static void sdp_query_rfcomm_service_response(uint8_t status){
18912381011SMilanka Ringwald     sdp_query_rfcomm_service_event_t service_event = {
19012381011SMilanka Ringwald         SDP_QUERY_RFCOMM_SERVICE,
19112381011SMilanka Ringwald         sdp_rfcomm_channel_nr,
19212381011SMilanka Ringwald         (uint8_t *)sdp_rfcomm_service_name
19312381011SMilanka Ringwald     };
19412381011SMilanka Ringwald     (*registered_sdp_app_callback)((sdp_query_event_t*)&service_event, registered_sdp_app_context);
19512381011SMilanka Ringwald }
19612381011SMilanka Ringwald 
19712381011SMilanka Ringwald void sdp_query_rfcomm_channel_and_name_for_uuid(bd_addr_t remote, uint16_t uuid){
19812381011SMilanka Ringwald 	// printf("sdp_query_rfcomm_channel_and_name_for_uuid %p\n", registered_sdp_app_callback);
19912381011SMilanka Ringwald 	sdp_query_rfcomm_service_response(0);
20012381011SMilanka Ringwald 	sdp_query_complete_response(0);
20112381011SMilanka Ringwald }
20212381011SMilanka Ringwald 
2036f1de21cSMilanka Ringwald 
20412381011SMilanka Ringwald void rfcomm_create_channel_internal(void * connection, bd_addr_t addr, uint8_t channel){
20512381011SMilanka Ringwald 	// RFCOMM_EVENT_OPEN_CHANNEL_COMPLETE
20612381011SMilanka Ringwald 	active_connection = connection;
20712381011SMilanka Ringwald     uint8_t event[16];
20812381011SMilanka Ringwald     uint8_t pos = 0;
20912381011SMilanka Ringwald     event[pos++] = RFCOMM_EVENT_OPEN_CHANNEL_COMPLETE;
21012381011SMilanka Ringwald     event[pos++] = sizeof(event) - 2;
21112381011SMilanka Ringwald     event[pos++] = 0;
21212381011SMilanka Ringwald 
2138c72a38bSMilanka Ringwald     bt_flip_addr(&event[pos], addr);
2148c72a38bSMilanka Ringwald     memcpy(dev_addr, addr, 6);
2158c72a38bSMilanka Ringwald     pos += 6;
2166f1de21cSMilanka Ringwald 
21712381011SMilanka Ringwald     bt_store_16(event,  pos, 1);   pos += 2;
21812381011SMilanka Ringwald 	event[pos++] = 0;
21912381011SMilanka Ringwald 
22012381011SMilanka Ringwald 	bt_store_16(event, pos, rfcomm_cid); pos += 2;       // channel ID
22112381011SMilanka Ringwald 	bt_store_16(event, pos, 200); pos += 2;   // max frame size
2228c72a38bSMilanka Ringwald     (*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, pos);
22312381011SMilanka Ringwald }
22412381011SMilanka Ringwald 
225d86bb743SMilanka Ringwald int rfcomm_can_send_packet_now(uint16_t rfcomm_cid){
226d86bb743SMilanka Ringwald 	return 1;
227d86bb743SMilanka Ringwald }
22812381011SMilanka Ringwald 
22912381011SMilanka Ringwald void rfcomm_disconnect_internal(uint16_t rfcomm_cid){
2309280a6ccSMilanka Ringwald 	uint8_t event[4];
2319280a6ccSMilanka Ringwald 	event[0] = RFCOMM_EVENT_CHANNEL_CLOSED;
2329280a6ccSMilanka Ringwald     event[1] = sizeof(event) - 2;
2339280a6ccSMilanka Ringwald     bt_store_16(event, 2, rfcomm_cid);
2349280a6ccSMilanka Ringwald     (*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, (uint8_t *) event, sizeof(event));
23512381011SMilanka Ringwald }
23612381011SMilanka Ringwald 
23712381011SMilanka Ringwald void rfcomm_register_packet_handler(void (*handler)(void * connection, uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size)){
23812381011SMilanka Ringwald 	registered_rfcomm_packet_handler = handler;
23912381011SMilanka Ringwald }
24012381011SMilanka Ringwald 
24112381011SMilanka Ringwald void rfcomm_register_service_internal(void * connection, uint8_t channel, uint16_t max_frame_size){
24212381011SMilanka Ringwald 	printf("rfcomm_register_service_internal\n");
24312381011SMilanka Ringwald }
24412381011SMilanka Ringwald 
24512381011SMilanka Ringwald 
24612381011SMilanka Ringwald void sdp_query_rfcomm_channel_and_name_for_search_pattern(bd_addr_t remote, uint8_t * des_serviceSearchPattern){
24712381011SMilanka Ringwald 	printf("sdp_query_rfcomm_channel_and_name_for_search_pattern\n");
24812381011SMilanka Ringwald }
24912381011SMilanka Ringwald 
25012381011SMilanka Ringwald 
25112381011SMilanka Ringwald void rfcomm_accept_connection_internal(uint16_t rfcomm_cid){
25212381011SMilanka Ringwald 	printf("rfcomm_accept_connection_internal \n");
25312381011SMilanka Ringwald }
25412381011SMilanka Ringwald 
255f78ff2fbSMilanka Ringwald void run_loop_add_timer(timer_source_t *timer){
256f78ff2fbSMilanka Ringwald }
257f78ff2fbSMilanka Ringwald 
258f78ff2fbSMilanka Ringwald int  run_loop_remove_timer(timer_source_t *timer){
259f78ff2fbSMilanka Ringwald     return 0;
260f78ff2fbSMilanka Ringwald }
261f78ff2fbSMilanka Ringwald void run_loop_set_timer_handler(timer_source_t *ts, void (*process)(timer_source_t *_ts)){
262f78ff2fbSMilanka Ringwald }
263f78ff2fbSMilanka Ringwald 
264f78ff2fbSMilanka Ringwald void run_loop_set_timer(timer_source_t *a, uint32_t timeout_in_ms){
265f78ff2fbSMilanka Ringwald }
266f78ff2fbSMilanka Ringwald 
267f78ff2fbSMilanka Ringwald 
2686f1de21cSMilanka Ringwald void hci_emit_disconnection_complete(uint16_t handle, uint8_t reason){
2696f1de21cSMilanka Ringwald     uint8_t event[6];
2706f1de21cSMilanka Ringwald     event[0] = HCI_EVENT_DISCONNECTION_COMPLETE;
2716f1de21cSMilanka Ringwald     event[1] = sizeof(event) - 2;
2726f1de21cSMilanka Ringwald     event[2] = 0; // status = OK
2736f1de21cSMilanka Ringwald     bt_store_16(event, 3, handle);
2746f1de21cSMilanka Ringwald     event[5] = reason;
2758c72a38bSMilanka Ringwald     (*registered_rfcomm_packet_handler)(active_connection, HCI_EVENT_PACKET, 0, event, sizeof(event));
2766f1de21cSMilanka Ringwald }
2776f1de21cSMilanka Ringwald 
2786f1de21cSMilanka Ringwald le_command_status_t gap_disconnect(hci_con_handle_t handle){
2796f1de21cSMilanka Ringwald     hci_emit_disconnection_complete(handle, 0);
2806f1de21cSMilanka Ringwald     return BLE_PERIPHERAL_OK;
2816f1de21cSMilanka Ringwald }
2826f1de21cSMilanka Ringwald 
283dd5554ccSMilanka Ringwald uint16_t hci_get_sco_voice_setting(){
284dd5554ccSMilanka Ringwald     return 0x40;
285dd5554ccSMilanka Ringwald }
286dc5611fdSMilanka Ringwald 
287*b51c851cSMatthias Ringwald int hci_remote_eSCO_supported(hci_con_handle_t handle){
288*b51c851cSMatthias Ringwald     return 0;
289*b51c851cSMatthias Ringwald }
290*b51c851cSMatthias Ringwald 
291656f7353SMilanka Ringwald void inject_rfcomm_command_to_hf(uint8_t * data, int len){
292656f7353SMilanka Ringwald     if (memcmp((char*)data, "AT", 2) == 0) return;
293656f7353SMilanka Ringwald 
294656f7353SMilanka Ringwald     prepare_rfcomm_buffer(data, len);
295c26caa21SMilanka Ringwald     if (data[0] == '+' || (data[0] == 'O' && data[1] == 'K')){
296c26caa21SMilanka Ringwald         printf("Send cmd to HF state machine: %s\n", data);
297656f7353SMilanka Ringwald     } else {
298c26caa21SMilanka Ringwald         printf("Trigger HF state machine - %s", data);
299656f7353SMilanka Ringwald     }
300656f7353SMilanka Ringwald     (*registered_rfcomm_packet_handler)(active_connection, RFCOMM_DATA_PACKET, rfcomm_cid, (uint8_t *) &rfcomm_payload[0], rfcomm_payload_len);
301656f7353SMilanka Ringwald }
302656f7353SMilanka Ringwald 
303656f7353SMilanka Ringwald void inject_rfcomm_command_to_ag(uint8_t * data, int len){
304656f7353SMilanka Ringwald     if (data[0] == '+') return;
305656f7353SMilanka Ringwald 
306656f7353SMilanka Ringwald     prepare_rfcomm_buffer(data, len);
307656f7353SMilanka Ringwald     if (memcmp((char*)data, "AT", 2) == 0){
30862656e33SMilanka Ringwald         printf("Send cmd to AG state machine: %s\n", data);
309656f7353SMilanka Ringwald     } else {
310c26caa21SMilanka Ringwald         printf("Trigger AG state machine - %s", data);
311656f7353SMilanka Ringwald     }
312656f7353SMilanka Ringwald     (*registered_rfcomm_packet_handler)(active_connection, RFCOMM_DATA_PACKET, rfcomm_cid, (uint8_t *) &rfcomm_payload[0], rfcomm_payload_len);
313656f7353SMilanka Ringwald }
314dc5611fdSMilanka Ringwald 
315dc5611fdSMilanka Ringwald 
316dc5611fdSMilanka Ringwald 
317dc5611fdSMilanka Ringwald 
318