* Fix TextBox word wrap behavior * Wrap width is 120 pixels, not 140. (140 is larger than the screen!) * Glyph width already includes spacing; don't add 1 additional px * When starting a new line, include wrapped glyph width in new line_width. * Call canvas_set_font before text_box_insert_endline so that glyph width is calculated using correct font. Previous approach worked somewhat well using default TextBoxFontText but this version is more robust, particularly when using TextBoxFontHex. * Add basic Mifare DESFire reading, file/app browser * Fix build with APP_ARCHIVE=0 * Add bool type to flipper_format * Add ability to save and load DESFire card data * Skip over NfcSceneDeviceInfo when viewing saved DESFire info * mf_df_clear: don't leak master key settings key versions * When opening a DESFire card from Archive, retain UID emulation behavior * rm unnecessary \r\n * show Popup instead of leaving view in bad state * Move NfcReaderRequestData out of union This makes it safe to emulate DESFire/EMV without clobbering card data. * Display saved DESFire cards via NfcSceneDeviceInfo * Display and save file metadata even when contents are missing This can happen when a file doesn't allow unauthenticated reads (see the call to mf_df_parse_read_data_response in nfc_worker.c). Co-authored-by: Kevin Wallace <git+flipperzero@kevin.wallace.seattle.wa.us> Co-authored-by: あく <alleteam@gmail.com> Co-authored-by: gornekich <n.gorbadey@gmail.com>
		
			
				
	
	
		
			165 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #pragma once
 | |
| 
 | |
| #include <m-string.h>
 | |
| #include <stdint.h>
 | |
| #include <stdbool.h>
 | |
| 
 | |
| #define MF_DF_GET_VERSION (0x60)
 | |
| #define MF_DF_GET_FREE_MEMORY (0x6E)
 | |
| #define MF_DF_GET_KEY_SETTINGS (0x45)
 | |
| #define MF_DF_GET_KEY_VERSION (0x64)
 | |
| #define MF_DF_GET_APPLICATION_IDS (0x6A)
 | |
| #define MF_DF_SELECT_APPLICATION (0x5A)
 | |
| #define MF_DF_GET_FILE_IDS (0x6F)
 | |
| #define MF_DF_GET_FILE_SETTINGS (0xF5)
 | |
| 
 | |
| #define MF_DF_READ_DATA (0xBD)
 | |
| #define MF_DF_GET_VALUE (0x6C)
 | |
| #define MF_DF_READ_RECORDS (0xBB)
 | |
| 
 | |
| typedef struct {
 | |
|     uint8_t hw_vendor;
 | |
|     uint8_t hw_type;
 | |
|     uint8_t hw_subtype;
 | |
|     uint8_t hw_major;
 | |
|     uint8_t hw_minor;
 | |
|     uint8_t hw_storage;
 | |
|     uint8_t hw_proto;
 | |
| 
 | |
|     uint8_t sw_vendor;
 | |
|     uint8_t sw_type;
 | |
|     uint8_t sw_subtype;
 | |
|     uint8_t sw_major;
 | |
|     uint8_t sw_minor;
 | |
|     uint8_t sw_storage;
 | |
|     uint8_t sw_proto;
 | |
| 
 | |
|     uint8_t uid[7];
 | |
|     uint8_t batch[5];
 | |
|     uint8_t prod_week;
 | |
|     uint8_t prod_year;
 | |
| } MifareDesfireVersion;
 | |
| 
 | |
| typedef struct {
 | |
|     uint32_t bytes;
 | |
| } MifareDesfireFreeMemory; // EV1+ only
 | |
| 
 | |
| typedef struct MifareDesfireKeyVersion {
 | |
|     uint8_t id;
 | |
|     uint8_t version;
 | |
|     struct MifareDesfireKeyVersion* next;
 | |
| } MifareDesfireKeyVersion;
 | |
| 
 | |
| typedef struct {
 | |
|     uint8_t change_key_id;
 | |
|     bool config_changeable;
 | |
|     bool free_create_delete;
 | |
|     bool free_directory_list;
 | |
|     bool master_key_changeable;
 | |
|     uint8_t max_keys;
 | |
|     MifareDesfireKeyVersion* key_version_head;
 | |
| } MifareDesfireKeySettings;
 | |
| 
 | |
| typedef enum {
 | |
|     MifareDesfireFileTypeStandard = 0,
 | |
|     MifareDesfireFileTypeBackup = 1,
 | |
|     MifareDesfireFileTypeValue = 2,
 | |
|     MifareDesfireFileTypeLinearRecord = 3,
 | |
|     MifareDesfireFileTypeCyclicRecord = 4,
 | |
| } MifareDesfireFileType;
 | |
| 
 | |
| typedef enum {
 | |
|     MifareDesfireFileCommunicationSettingsPlaintext = 0,
 | |
|     MifareDesfireFileCommunicationSettingsAuthenticated = 1,
 | |
|     MifareDesfireFileCommunicationSettingsEnciphered = 3,
 | |
| } MifareDesfireFileCommunicationSettings;
 | |
| 
 | |
| typedef struct MifareDesfireFile {
 | |
|     uint8_t id;
 | |
|     MifareDesfireFileType type;
 | |
|     MifareDesfireFileCommunicationSettings comm;
 | |
|     uint16_t access_rights;
 | |
|     union {
 | |
|         struct {
 | |
|             uint32_t size;
 | |
|         } data;
 | |
|         struct {
 | |
|             uint32_t lo_limit;
 | |
|             uint32_t hi_limit;
 | |
|             uint32_t limited_credit_value;
 | |
|             bool limited_credit_enabled;
 | |
|         } value;
 | |
|         struct {
 | |
|             uint32_t size;
 | |
|             uint32_t max;
 | |
|             uint32_t cur;
 | |
|         } record;
 | |
|     } settings;
 | |
|     uint8_t* contents;
 | |
| 
 | |
|     struct MifareDesfireFile* next;
 | |
| } MifareDesfireFile;
 | |
| 
 | |
| typedef struct MifareDesfireApplication {
 | |
|     uint8_t id[3];
 | |
|     MifareDesfireKeySettings* key_settings;
 | |
|     MifareDesfireFile* file_head;
 | |
| 
 | |
|     struct MifareDesfireApplication* next;
 | |
| } MifareDesfireApplication;
 | |
| 
 | |
| typedef struct {
 | |
|     MifareDesfireVersion version;
 | |
|     MifareDesfireFreeMemory* free_memory;
 | |
|     MifareDesfireKeySettings* master_key_settings;
 | |
|     MifareDesfireApplication* app_head;
 | |
| } MifareDesfireData;
 | |
| 
 | |
| void mf_df_clear(MifareDesfireData* data);
 | |
| 
 | |
| void mf_df_cat_data(MifareDesfireData* data, string_t out);
 | |
| void mf_df_cat_card_info(MifareDesfireData* data, string_t out);
 | |
| void mf_df_cat_version(MifareDesfireVersion* version, string_t out);
 | |
| void mf_df_cat_free_mem(MifareDesfireFreeMemory* free_mem, string_t out);
 | |
| void mf_df_cat_key_settings(MifareDesfireKeySettings* ks, string_t out);
 | |
| void mf_df_cat_application_info(MifareDesfireApplication* app, string_t out);
 | |
| void mf_df_cat_application(MifareDesfireApplication* app, string_t out);
 | |
| void mf_df_cat_file(MifareDesfireFile* file, string_t out);
 | |
| 
 | |
| bool mf_df_check_card_type(uint8_t ATQA0, uint8_t ATQA1, uint8_t SAK);
 | |
| 
 | |
| uint16_t mf_df_prepare_get_version(uint8_t* dest);
 | |
| bool mf_df_parse_get_version_response(uint8_t* buf, uint16_t len, MifareDesfireVersion* out);
 | |
| 
 | |
| uint16_t mf_df_prepare_get_free_memory(uint8_t* dest);
 | |
| bool mf_df_parse_get_free_memory_response(uint8_t* buf, uint16_t len, MifareDesfireFreeMemory* out);
 | |
| 
 | |
| uint16_t mf_df_prepare_get_key_settings(uint8_t* dest);
 | |
| bool mf_df_parse_get_key_settings_response(
 | |
|     uint8_t* buf,
 | |
|     uint16_t len,
 | |
|     MifareDesfireKeySettings* out);
 | |
| 
 | |
| uint16_t mf_df_prepare_get_key_version(uint8_t* dest, uint8_t key_id);
 | |
| bool mf_df_parse_get_key_version_response(uint8_t* buf, uint16_t len, MifareDesfireKeyVersion* out);
 | |
| 
 | |
| uint16_t mf_df_prepare_get_application_ids(uint8_t* dest);
 | |
| bool mf_df_parse_get_application_ids_response(
 | |
|     uint8_t* buf,
 | |
|     uint16_t len,
 | |
|     MifareDesfireApplication** app_head);
 | |
| 
 | |
| uint16_t mf_df_prepare_select_application(uint8_t* dest, uint8_t id[3]);
 | |
| bool mf_df_parse_select_application_response(uint8_t* buf, uint16_t len);
 | |
| 
 | |
| uint16_t mf_df_prepare_get_file_ids(uint8_t* dest);
 | |
| bool mf_df_parse_get_file_ids_response(uint8_t* buf, uint16_t len, MifareDesfireFile** file_head);
 | |
| 
 | |
| uint16_t mf_df_prepare_get_file_settings(uint8_t* dest, uint8_t file_id);
 | |
| bool mf_df_parse_get_file_settings_response(uint8_t* buf, uint16_t len, MifareDesfireFile* out);
 | |
| 
 | |
| uint16_t mf_df_prepare_read_data(uint8_t* dest, uint8_t file_id, uint32_t offset, uint32_t len);
 | |
| uint16_t mf_df_prepare_get_value(uint8_t* dest, uint8_t file_id);
 | |
| uint16_t mf_df_prepare_read_records(uint8_t* dest, uint8_t file_id, uint32_t offset, uint32_t len);
 | |
| bool mf_df_parse_read_data_response(uint8_t* buf, uint16_t len, MifareDesfireFile* out);
 |