mirror of
https://github.com/MariaDB/server.git
synced 2025-01-18 13:02:28 +01:00
a10ae35328
Essentially, the problem is that safemalloc is excruciatingly slow as it checks all allocated blocks for overrun at each memory management primitive, yielding a almost exponential slowdown for the memory management functions (malloc, realloc, free). The overrun check basically consists of verifying some bytes of a block for certain magic keys, which catches some simple forms of overrun. Another minor problem is violation of aliasing rules and that its own internal list of blocks is prone to corruption. Another issue with safemalloc is rather the maintenance cost as the tool has a significant impact on the server code. Given the magnitude of memory debuggers available nowadays, especially those that are provided with the platform malloc implementation, maintenance of a in-house and largely obsolete memory debugger becomes a burden that is not worth the effort due to its slowness and lack of support for detecting more common forms of heap corruption. Since there are third-party tools that can provide the same functionality at a lower or comparable performance cost, the solution is to simply remove safemalloc. Third-party tools can provide the same functionality at a lower or comparable performance cost. The removal of safemalloc also allows a simplification of the malloc wrappers, removing quite a bit of kludge: redefinition of my_malloc, my_free and the removal of the unused second argument of my_free. Since free() always check whether the supplied pointer is null, redudant checks are also removed. Also, this patch adds unit testing for my_malloc and moves my_realloc implementation into the same file as the other memory allocation primitives.
223 lines
6.3 KiB
C
223 lines
6.3 KiB
C
/* Copyright (C) 2000-2005 MySQL AB
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
#include "mysys_priv.h"
|
|
#include "m_string.h"
|
|
#ifdef __WIN__
|
|
|
|
/* Windows NT/2000 discretionary access control utility functions. */
|
|
|
|
/*
|
|
Check if the operating system is built on NT technology.
|
|
|
|
RETURN
|
|
0 Windows 95/98/Me
|
|
1 otherwise
|
|
*/
|
|
|
|
static my_bool is_nt()
|
|
{
|
|
return GetVersion() < 0x80000000;
|
|
}
|
|
|
|
/*
|
|
Auxilary structure to store pointers to the data which we need to keep
|
|
around while SECURITY_ATTRIBUTES is in use.
|
|
*/
|
|
|
|
typedef struct st_my_security_attr
|
|
{
|
|
PSID everyone_sid;
|
|
PACL dacl;
|
|
} My_security_attr;
|
|
|
|
|
|
/*
|
|
Allocate and initialize SECURITY_ATTRIBUTES setting up access
|
|
rights for the owner and group `Everybody'.
|
|
|
|
SYNOPSIS
|
|
my_security_attr_create()
|
|
psa [OUT] pointer to store the pointer to SA in
|
|
perror [OUT] pointer to store error message if there was an
|
|
error
|
|
owner_rights [IN] access rights for the owner
|
|
everyone_rights [IN] access rights for group Everybody
|
|
|
|
DESCRIPTION
|
|
Set up the security attributes to provide clients with sufficient
|
|
access rights to a kernel object. We need this function
|
|
because if we simply grant all access to everybody (by installing
|
|
a NULL DACL) a mailicious user can attempt a denial of service
|
|
attack by taking ownership over the kernel object. Upon successful
|
|
return `psa' contains a pointer to SECUIRITY_ATTRIBUTES that can be used
|
|
to create kernel objects with proper access rights.
|
|
|
|
RETURN
|
|
0 success, psa is 0 or points to a valid SA structure,
|
|
perror is left intact
|
|
!0 error, SA is set to 0, error message is stored in perror
|
|
*/
|
|
|
|
int my_security_attr_create(SECURITY_ATTRIBUTES **psa, const char **perror,
|
|
DWORD owner_rights, DWORD everyone_rights)
|
|
{
|
|
/* Top-level SID authority */
|
|
SID_IDENTIFIER_AUTHORITY world_auth= SECURITY_WORLD_SID_AUTHORITY;
|
|
PSID everyone_sid= 0;
|
|
HANDLE htoken= 0;
|
|
SECURITY_ATTRIBUTES *sa= 0;
|
|
PACL dacl= 0;
|
|
DWORD owner_token_length, dacl_length;
|
|
SECURITY_DESCRIPTOR *sd;
|
|
PTOKEN_USER owner_token;
|
|
PSID owner_sid;
|
|
My_security_attr *attr;
|
|
|
|
if (! is_nt())
|
|
{
|
|
*psa= 0;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
Get SID of Everyone group. Easier to retrieve all SIDs each time
|
|
this function is called than worry about thread safety.
|
|
*/
|
|
if (! AllocateAndInitializeSid(&world_auth, 1, SECURITY_WORLD_RID,
|
|
0, 0, 0, 0, 0, 0, 0, &everyone_sid))
|
|
{
|
|
*perror= "Failed to retrieve the SID of Everyone group";
|
|
goto error;
|
|
}
|
|
|
|
/*
|
|
Get SID of the owner. Using GetSecurityInfo this task can be done
|
|
in just one call instead of five, but GetSecurityInfo declared in
|
|
aclapi.h, so I hesitate to use it.
|
|
SIC: OpenThreadToken works only if there is an active impersonation
|
|
token, hence OpenProcessToken is used.
|
|
*/
|
|
if (! OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, &htoken))
|
|
{
|
|
*perror= "Failed to retrieve thread access token";
|
|
goto error;
|
|
}
|
|
GetTokenInformation(htoken, TokenUser, 0, 0, &owner_token_length);
|
|
|
|
if (! my_multi_malloc(MYF(MY_WME),
|
|
&sa, ALIGN_SIZE(sizeof(SECURITY_ATTRIBUTES)) +
|
|
sizeof(My_security_attr),
|
|
&sd, sizeof(SECURITY_DESCRIPTOR),
|
|
&owner_token, owner_token_length,
|
|
0))
|
|
{
|
|
*perror= "Failed to allocate memory for SECURITY_ATTRIBUTES";
|
|
goto error;
|
|
}
|
|
bzero(owner_token, owner_token_length);
|
|
if (! GetTokenInformation(htoken, TokenUser, owner_token,
|
|
owner_token_length, &owner_token_length))
|
|
{
|
|
*perror= "GetTokenInformation failed";
|
|
goto error;
|
|
}
|
|
owner_sid= owner_token->User.Sid;
|
|
|
|
if (! IsValidSid(owner_sid))
|
|
{
|
|
*perror= "IsValidSid failed";
|
|
goto error;
|
|
}
|
|
|
|
/* Calculate the amount of memory that must be allocated for the DACL */
|
|
dacl_length= sizeof(ACL) + (sizeof(ACCESS_ALLOWED_ACE)-sizeof(DWORD)) * 2 +
|
|
GetLengthSid(everyone_sid) + GetLengthSid(owner_sid);
|
|
|
|
/* Create an ACL */
|
|
if (! (dacl= (PACL) my_malloc(dacl_length, MYF(MY_ZEROFILL|MY_WME))))
|
|
{
|
|
*perror= "Failed to allocate memory for DACL";
|
|
goto error;
|
|
}
|
|
if (! InitializeAcl(dacl, dacl_length, ACL_REVISION))
|
|
{
|
|
*perror= "Failed to initialize DACL";
|
|
goto error;
|
|
}
|
|
if (! AddAccessAllowedAce(dacl, ACL_REVISION, everyone_rights, everyone_sid))
|
|
{
|
|
*perror= "Failed to set up DACL";
|
|
goto error;
|
|
}
|
|
if (! AddAccessAllowedAce(dacl, ACL_REVISION, owner_rights, owner_sid))
|
|
{
|
|
*perror= "Failed to set up DACL";
|
|
goto error;
|
|
}
|
|
if (! InitializeSecurityDescriptor(sd, SECURITY_DESCRIPTOR_REVISION))
|
|
{
|
|
*perror= "Could not initialize security descriptor";
|
|
goto error;
|
|
}
|
|
if (! SetSecurityDescriptorDacl(sd, TRUE, dacl, FALSE))
|
|
{
|
|
*perror= "Failed to install DACL";
|
|
goto error;
|
|
}
|
|
|
|
sa->nLength= sizeof(*sa);
|
|
sa->bInheritHandle= TRUE;
|
|
sa->lpSecurityDescriptor= sd;
|
|
/* Save pointers to everyone_sid and dacl to be able to clean them up */
|
|
attr= (My_security_attr*) (((char*) sa) + ALIGN_SIZE(sizeof(*sa)));
|
|
attr->everyone_sid= everyone_sid;
|
|
attr->dacl= dacl;
|
|
*psa= sa;
|
|
|
|
CloseHandle(htoken);
|
|
return 0;
|
|
error:
|
|
if (everyone_sid)
|
|
FreeSid(everyone_sid);
|
|
if (htoken)
|
|
CloseHandle(htoken);
|
|
my_free(sa);
|
|
my_free(dacl);
|
|
*psa= 0;
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
Cleanup security attributes freeing used memory.
|
|
|
|
SYNOPSIS
|
|
my_security_attr_free()
|
|
sa security attributes
|
|
*/
|
|
|
|
void my_security_attr_free(SECURITY_ATTRIBUTES *sa)
|
|
{
|
|
if (sa)
|
|
{
|
|
My_security_attr *attr= (My_security_attr*)
|
|
(((char*)sa) + ALIGN_SIZE(sizeof(*sa)));
|
|
FreeSid(attr->everyone_sid);
|
|
my_free(attr->dacl);
|
|
my_free(sa);
|
|
}
|
|
}
|
|
|
|
#endif /* __WIN__ */
|