blob: aa88fc1a664a65b6542c1b2bd1edc314a69b49f0 [file] [log] [blame]
Paul Bakker747a83a2014-02-01 22:50:07 +01001/*
2 * Platform abstraction layer
3 *
Paul Bakkere021a4b2016-06-01 11:25:44 +01004 * Copyright (C) 2006-2016, ARM Limited, All Rights Reserved
Manuel Pégourié-Gonnard37ff1402015-09-04 14:21:07 +02005 * SPDX-License-Identifier: Apache-2.0
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License"); you may
8 * not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
15 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
Paul Bakker747a83a2014-02-01 22:50:07 +010018 *
Manuel Pégourié-Gonnardfe446432015-03-06 13:17:10 +000019 * This file is part of mbed TLS (https://tls.mbed.org)
Paul Bakker747a83a2014-02-01 22:50:07 +010020 */
21
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +020022#if !defined(MBEDTLS_CONFIG_FILE)
Manuel Pégourié-Gonnard7f809972015-03-09 17:05:11 +000023#include "mbedtls/config.h"
Manuel Pégourié-Gonnardcef4ad22014-04-29 12:39:06 +020024#else
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +020025#include MBEDTLS_CONFIG_FILE
Manuel Pégourié-Gonnardcef4ad22014-04-29 12:39:06 +020026#endif
Paul Bakker747a83a2014-02-01 22:50:07 +010027
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +020028#if defined(MBEDTLS_PLATFORM_C)
Paul Bakker747a83a2014-02-01 22:50:07 +010029
Manuel Pégourié-Gonnard7f809972015-03-09 17:05:11 +000030#include "mbedtls/platform.h"
Paul Bakker747a83a2014-02-01 22:50:07 +010031
Manuel Pégourié-Gonnarda268da92017-12-20 12:52:49 +010032#if defined(MBEDTLS_ENTROPY_NV_SEED) && \
33 !defined(MBEDTLS_PLATFORM_NO_STD_FUNCTIONS) && defined(MBEDTLS_FS_IO)
34/* Implementation that should never be optimized out by the compiler */
35static void mbedtls_zeroize( void *v, size_t n ) {
36 volatile unsigned char *p = (unsigned char*)v; while( n-- ) *p++ = 0;
37}
38#endif
39
Hanno Becker643f3112018-10-11 10:26:55 +010040/* The compile time configuration of memory allocation via the macros
41 * MBEDTLS_PLATFORM_{FREE/CALLOC}_MACRO takes precedence over the runtime
42 * configuration via mbedtls_platform_set_calloc_free(). So, omit everything
43 * related to the latter if MBEDTLS_PLATFORM_{FREE/CALLOC}_MACRO are defined. */
44#if defined(MBEDTLS_PLATFORM_MEMORY) && \
45 !( defined(MBEDTLS_PLATFORM_CALLOC_MACRO) && \
46 defined(MBEDTLS_PLATFORM_FREE_MACRO) )
47
Manuel Pégourié-Gonnardb9ef1182015-05-26 16:15:20 +020048#if !defined(MBEDTLS_PLATFORM_STD_CALLOC)
49static void *platform_calloc_uninit( size_t n, size_t size )
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010050{
Manuel Pégourié-Gonnardb9ef1182015-05-26 16:15:20 +020051 ((void) n);
52 ((void) size);
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010053 return( NULL );
54}
55
Manuel Pégourié-Gonnardb9ef1182015-05-26 16:15:20 +020056#define MBEDTLS_PLATFORM_STD_CALLOC platform_calloc_uninit
57#endif /* !MBEDTLS_PLATFORM_STD_CALLOC */
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010058
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +020059#if !defined(MBEDTLS_PLATFORM_STD_FREE)
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010060static void platform_free_uninit( void *ptr )
61{
62 ((void) ptr);
63}
64
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +020065#define MBEDTLS_PLATFORM_STD_FREE platform_free_uninit
66#endif /* !MBEDTLS_PLATFORM_STD_FREE */
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010067
Manuel Pégourié-Gonnardb9ef1182015-05-26 16:15:20 +020068void * (*mbedtls_calloc)( size_t, size_t ) = MBEDTLS_PLATFORM_STD_CALLOC;
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +020069void (*mbedtls_free)( void * ) = MBEDTLS_PLATFORM_STD_FREE;
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010070
Manuel Pégourié-Gonnardb9ef1182015-05-26 16:15:20 +020071int mbedtls_platform_set_calloc_free( void * (*calloc_func)( size_t, size_t ),
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010072 void (*free_func)( void * ) )
73{
Manuel Pégourié-Gonnardb9ef1182015-05-26 16:15:20 +020074 mbedtls_calloc = calloc_func;
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +020075 mbedtls_free = free_func;
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010076 return( 0 );
77}
Hanno Becker643f3112018-10-11 10:26:55 +010078#endif /* MBEDTLS_PLATFORM_MEMORY &&
79 !( defined(MBEDTLS_PLATFORM_CALLOC_MACRO) &&
80 defined(MBEDTLS_PLATFORM_FREE_MACRO) ) */
Paul Bakkerdefc0ca2014-02-04 17:30:24 +010081
Manuel Pégourié-Gonnard6c0c8e02015-06-22 10:23:34 +020082#if defined(_WIN32)
83#include <stdarg.h>
84int mbedtls_platform_win32_snprintf( char *s, size_t n, const char *fmt, ... )
85{
86 int ret;
87 va_list argp;
88
Manuel Pégourié-Gonnardf659d2c2015-06-26 17:45:00 +020089 /* Avoid calling the invalid parameter handler by checking ourselves */
90 if( s == NULL || n == 0 || fmt == NULL )
91 return( -1 );
92
Manuel Pégourié-Gonnard6c0c8e02015-06-22 10:23:34 +020093 va_start( argp, fmt );
Ron Eldorbc18eb32017-09-06 17:49:10 +030094#if defined(_TRUNCATE) && !defined(__MINGW32__)
Manuel Pégourié-Gonnard6c0c8e02015-06-22 10:23:34 +020095 ret = _vsnprintf_s( s, n, _TRUNCATE, fmt, argp );
Manuel Pégourié-Gonnarda4f055f2015-07-08 16:46:13 +020096#else
97 ret = _vsnprintf( s, n, fmt, argp );
98 if( ret < 0 || (size_t) ret == n )
99 {
100 s[n-1] = '\0';
101 ret = -1;
102 }
103#endif
Manuel Pégourié-Gonnard6c0c8e02015-06-22 10:23:34 +0200104 va_end( argp );
105
106 return( ret );
107}
108#endif
109
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200110#if defined(MBEDTLS_PLATFORM_SNPRINTF_ALT)
111#if !defined(MBEDTLS_PLATFORM_STD_SNPRINTF)
Rich Evans46b0a8d2015-01-30 10:47:32 +0000112/*
113 * Make dummy function to prevent NULL pointer dereferences
114 */
115static int platform_snprintf_uninit( char * s, size_t n,
116 const char * format, ... )
117{
118 ((void) s);
119 ((void) n);
Manuel Pégourié-Gonnarddccb80b2015-06-03 10:20:33 +0100120 ((void) format);
Rich Evans46b0a8d2015-01-30 10:47:32 +0000121 return( 0 );
122}
123
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200124#define MBEDTLS_PLATFORM_STD_SNPRINTF platform_snprintf_uninit
125#endif /* !MBEDTLS_PLATFORM_STD_SNPRINTF */
Rich Evans46b0a8d2015-01-30 10:47:32 +0000126
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200127int (*mbedtls_snprintf)( char * s, size_t n,
Rich Evans46b0a8d2015-01-30 10:47:32 +0000128 const char * format,
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200129 ... ) = MBEDTLS_PLATFORM_STD_SNPRINTF;
Rich Evans46b0a8d2015-01-30 10:47:32 +0000130
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200131int mbedtls_platform_set_snprintf( int (*snprintf_func)( char * s, size_t n,
Rich Evans46b0a8d2015-01-30 10:47:32 +0000132 const char * format,
133 ... ) )
134{
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200135 mbedtls_snprintf = snprintf_func;
Rich Evans46b0a8d2015-01-30 10:47:32 +0000136 return( 0 );
137}
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200138#endif /* MBEDTLS_PLATFORM_SNPRINTF_ALT */
Rich Evans46b0a8d2015-01-30 10:47:32 +0000139
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200140#if defined(MBEDTLS_PLATFORM_PRINTF_ALT)
141#if !defined(MBEDTLS_PLATFORM_STD_PRINTF)
Paul Bakker747a83a2014-02-01 22:50:07 +0100142/*
143 * Make dummy function to prevent NULL pointer dereferences
144 */
145static int platform_printf_uninit( const char *format, ... )
146{
147 ((void) format);
148 return( 0 );
149}
150
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200151#define MBEDTLS_PLATFORM_STD_PRINTF platform_printf_uninit
152#endif /* !MBEDTLS_PLATFORM_STD_PRINTF */
Paul Bakker747a83a2014-02-01 22:50:07 +0100153
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200154int (*mbedtls_printf)( const char *, ... ) = MBEDTLS_PLATFORM_STD_PRINTF;
Paul Bakker747a83a2014-02-01 22:50:07 +0100155
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200156int mbedtls_platform_set_printf( int (*printf_func)( const char *, ... ) )
Paul Bakker747a83a2014-02-01 22:50:07 +0100157{
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200158 mbedtls_printf = printf_func;
Paul Bakker747a83a2014-02-01 22:50:07 +0100159 return( 0 );
160}
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200161#endif /* MBEDTLS_PLATFORM_PRINTF_ALT */
Paul Bakker747a83a2014-02-01 22:50:07 +0100162
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200163#if defined(MBEDTLS_PLATFORM_FPRINTF_ALT)
164#if !defined(MBEDTLS_PLATFORM_STD_FPRINTF)
Paul Bakker747a83a2014-02-01 22:50:07 +0100165/*
166 * Make dummy function to prevent NULL pointer dereferences
167 */
168static int platform_fprintf_uninit( FILE *stream, const char *format, ... )
169{
170 ((void) stream);
171 ((void) format);
172 return( 0 );
173}
174
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200175#define MBEDTLS_PLATFORM_STD_FPRINTF platform_fprintf_uninit
176#endif /* !MBEDTLS_PLATFORM_STD_FPRINTF */
Paul Bakker747a83a2014-02-01 22:50:07 +0100177
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200178int (*mbedtls_fprintf)( FILE *, const char *, ... ) =
179 MBEDTLS_PLATFORM_STD_FPRINTF;
Paul Bakker747a83a2014-02-01 22:50:07 +0100180
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200181int mbedtls_platform_set_fprintf( int (*fprintf_func)( FILE *, const char *, ... ) )
Paul Bakker747a83a2014-02-01 22:50:07 +0100182{
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200183 mbedtls_fprintf = fprintf_func;
Paul Bakker747a83a2014-02-01 22:50:07 +0100184 return( 0 );
185}
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200186#endif /* MBEDTLS_PLATFORM_FPRINTF_ALT */
Paul Bakker747a83a2014-02-01 22:50:07 +0100187
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200188#if defined(MBEDTLS_PLATFORM_EXIT_ALT)
189#if !defined(MBEDTLS_PLATFORM_STD_EXIT)
Rich Evansc39cb492015-01-30 12:01:34 +0000190/*
191 * Make dummy function to prevent NULL pointer dereferences
192 */
193static void platform_exit_uninit( int status )
194{
195 ((void) status);
Rich Evansc39cb492015-01-30 12:01:34 +0000196}
197
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200198#define MBEDTLS_PLATFORM_STD_EXIT platform_exit_uninit
199#endif /* !MBEDTLS_PLATFORM_STD_EXIT */
Rich Evansc39cb492015-01-30 12:01:34 +0000200
Manuel Pégourié-Gonnard7ee5ddd2015-06-03 10:33:55 +0100201void (*mbedtls_exit)( int status ) = MBEDTLS_PLATFORM_STD_EXIT;
Rich Evansc39cb492015-01-30 12:01:34 +0000202
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200203int mbedtls_platform_set_exit( void (*exit_func)( int status ) )
Rich Evansc39cb492015-01-30 12:01:34 +0000204{
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200205 mbedtls_exit = exit_func;
Rich Evansc39cb492015-01-30 12:01:34 +0000206 return( 0 );
207}
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200208#endif /* MBEDTLS_PLATFORM_EXIT_ALT */
Rich Evansc39cb492015-01-30 12:01:34 +0000209
Simon Butcher23e97782016-07-13 13:31:08 +0100210#if defined(MBEDTLS_HAVE_TIME)
211
SimonBd5800b72016-04-26 07:43:27 +0100212#if defined(MBEDTLS_PLATFORM_TIME_ALT)
213#if !defined(MBEDTLS_PLATFORM_STD_TIME)
214/*
215 * Make dummy function to prevent NULL pointer dereferences
216 */
217static mbedtls_time_t platform_time_uninit( mbedtls_time_t* timer )
218{
219 ((void) timer);
Simon Butcher3fe6cd32016-04-26 19:51:29 +0100220 return( 0 );
SimonBd5800b72016-04-26 07:43:27 +0100221}
222
223#define MBEDTLS_PLATFORM_STD_TIME platform_time_uninit
224#endif /* !MBEDTLS_PLATFORM_STD_TIME */
225
Simon Butcher3fe6cd32016-04-26 19:51:29 +0100226mbedtls_time_t (*mbedtls_time)( mbedtls_time_t* timer ) = MBEDTLS_PLATFORM_STD_TIME;
SimonBd5800b72016-04-26 07:43:27 +0100227
Simon Butcher3fe6cd32016-04-26 19:51:29 +0100228int mbedtls_platform_set_time( mbedtls_time_t (*time_func)( mbedtls_time_t* timer ) )
SimonBd5800b72016-04-26 07:43:27 +0100229{
230 mbedtls_time = time_func;
231 return( 0 );
232}
233#endif /* MBEDTLS_PLATFORM_TIME_ALT */
234
Simon Butcher23e97782016-07-13 13:31:08 +0100235#endif /* MBEDTLS_HAVE_TIME */
236
Paul Bakkere021a4b2016-06-01 11:25:44 +0100237#if defined(MBEDTLS_ENTROPY_NV_SEED)
238#if !defined(MBEDTLS_PLATFORM_NO_STD_FUNCTIONS) && defined(MBEDTLS_FS_IO)
239/* Default implementations for the platform independent seed functions use
240 * standard libc file functions to read from and write to a pre-defined filename
241 */
242int mbedtls_platform_std_nv_seed_read( unsigned char *buf, size_t buf_len )
243{
244 FILE *file;
245 size_t n;
246
247 if( ( file = fopen( MBEDTLS_PLATFORM_STD_NV_SEED_FILE, "rb" ) ) == NULL )
Andres Amaya Garcia79a2e7e2017-06-26 11:10:22 +0100248 return( -1 );
Paul Bakkere021a4b2016-06-01 11:25:44 +0100249
250 if( ( n = fread( buf, 1, buf_len, file ) ) != buf_len )
251 {
252 fclose( file );
Andres Amaya Garcia79a2e7e2017-06-26 11:10:22 +0100253 mbedtls_zeroize( buf, buf_len );
254 return( -1 );
Paul Bakkere021a4b2016-06-01 11:25:44 +0100255 }
256
257 fclose( file );
Simon B3249cb72016-11-03 01:11:37 +0000258 return( (int)n );
Paul Bakkere021a4b2016-06-01 11:25:44 +0100259}
260
261int mbedtls_platform_std_nv_seed_write( unsigned char *buf, size_t buf_len )
262{
263 FILE *file;
264 size_t n;
265
266 if( ( file = fopen( MBEDTLS_PLATFORM_STD_NV_SEED_FILE, "w" ) ) == NULL )
267 return -1;
268
269 if( ( n = fwrite( buf, 1, buf_len, file ) ) != buf_len )
270 {
271 fclose( file );
272 return -1;
273 }
274
275 fclose( file );
Simon B3249cb72016-11-03 01:11:37 +0000276 return( (int)n );
Paul Bakkere021a4b2016-06-01 11:25:44 +0100277}
278#endif /* MBEDTLS_PLATFORM_NO_STD_FUNCTIONS */
279
280#if defined(MBEDTLS_PLATFORM_NV_SEED_ALT)
281#if !defined(MBEDTLS_PLATFORM_STD_NV_SEED_READ)
282/*
283 * Make dummy function to prevent NULL pointer dereferences
284 */
285static int platform_nv_seed_read_uninit( unsigned char *buf, size_t buf_len )
286{
287 ((void) buf);
288 ((void) buf_len);
289 return( -1 );
290}
291
292#define MBEDTLS_PLATFORM_STD_NV_SEED_READ platform_nv_seed_read_uninit
293#endif /* !MBEDTLS_PLATFORM_STD_NV_SEED_READ */
294
295#if !defined(MBEDTLS_PLATFORM_STD_NV_SEED_WRITE)
296/*
297 * Make dummy function to prevent NULL pointer dereferences
298 */
299static int platform_nv_seed_write_uninit( unsigned char *buf, size_t buf_len )
300{
301 ((void) buf);
302 ((void) buf_len);
303 return( -1 );
304}
305
306#define MBEDTLS_PLATFORM_STD_NV_SEED_WRITE platform_nv_seed_write_uninit
307#endif /* !MBEDTLS_PLATFORM_STD_NV_SEED_WRITE */
308
309int (*mbedtls_nv_seed_read)( unsigned char *buf, size_t buf_len ) =
310 MBEDTLS_PLATFORM_STD_NV_SEED_READ;
311int (*mbedtls_nv_seed_write)( unsigned char *buf, size_t buf_len ) =
312 MBEDTLS_PLATFORM_STD_NV_SEED_WRITE;
313
314int mbedtls_platform_set_nv_seed(
315 int (*nv_seed_read_func)( unsigned char *buf, size_t buf_len ),
316 int (*nv_seed_write_func)( unsigned char *buf, size_t buf_len ) )
317{
318 mbedtls_nv_seed_read = nv_seed_read_func;
319 mbedtls_nv_seed_write = nv_seed_write_func;
320 return( 0 );
321}
322#endif /* MBEDTLS_PLATFORM_NV_SEED_ALT */
323#endif /* MBEDTLS_ENTROPY_NV_SEED */
324
Andres Amaya Garciad91f99f2017-07-18 10:23:04 +0100325#if !defined(MBEDTLS_PLATFORM_SETUP_TEARDOWN_ALT)
Andres Amaya Garcia2a6f39c2017-07-07 13:03:23 +0100326/*
Andres Amaya Garcia3c8a39d2017-07-12 11:25:17 +0100327 * Placeholder platform setup that does nothing by default
Andres Amaya Garcia2a6f39c2017-07-07 13:03:23 +0100328 */
Andres Amaya Garcia3c8a39d2017-07-12 11:25:17 +0100329int mbedtls_platform_setup( mbedtls_platform_context *ctx )
Andres Amaya Garcia2a6f39c2017-07-07 13:03:23 +0100330{
331 (void)ctx;
332
333 return( 0 );
334}
335
336/*
Andres Amaya Garcia3c8a39d2017-07-12 11:25:17 +0100337 * Placeholder platform teardown that does nothing by default
Andres Amaya Garcia2a6f39c2017-07-07 13:03:23 +0100338 */
Andres Amaya Garcia3c8a39d2017-07-12 11:25:17 +0100339void mbedtls_platform_teardown( mbedtls_platform_context *ctx )
Andres Amaya Garcia2a6f39c2017-07-07 13:03:23 +0100340{
341 (void)ctx;
342}
Andres Amaya Garciad91f99f2017-07-18 10:23:04 +0100343#endif /* MBEDTLS_PLATFORM_SETUP_TEARDOWN_ALT */
Andres Amaya Garcia2a6f39c2017-07-07 13:03:23 +0100344
Manuel Pégourié-Gonnard2cf5a7c2015-04-08 12:49:31 +0200345#endif /* MBEDTLS_PLATFORM_C */