Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
MariaDB
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
MariaDB
Commits
5605117d
Commit
5605117d
authored
Aug 19, 2005
by
petr@mysql.com
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix for BUG#10957 "stop instance, issued after flush instances causes IM to crash"
Recommited with post-review fixes
parent
41c93e36
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
241 additions
and
101 deletions
+241
-101
server-tools/instance-manager/instance.cc
server-tools/instance-manager/instance.cc
+237
-99
server-tools/instance-manager/instance.h
server-tools/instance-manager/instance.h
+2
-2
server-tools/instance-manager/instance_options.cc
server-tools/instance-manager/instance_options.cc
+2
-0
No files found.
server-tools/instance-manager/instance.cc
View file @
5605117d
...
@@ -36,6 +36,16 @@
...
@@ -36,6 +36,16 @@
#include <m_string.h>
#include <m_string.h>
#include <mysql.h>
#include <mysql.h>
static
void
start_and_monitor_instance
(
Instance_options
*
old_instance_options
,
Instance_map
*
instance_map
);
#ifndef _WIN_
typedef
pid_t
My_process_info
;
#else
typedef
PROCESS_INFORMATION
My_process_info
;
#endif
C_MODE_START
C_MODE_START
/*
/*
...
@@ -48,13 +58,224 @@ C_MODE_START
...
@@ -48,13 +58,224 @@ C_MODE_START
pthread_handler_decl
(
proxy
,
arg
)
pthread_handler_decl
(
proxy
,
arg
)
{
{
Instance
*
instance
=
(
Instance
*
)
arg
;
Instance
*
instance
=
(
Instance
*
)
arg
;
instance
->
fork_and_monitor
();
start_and_monitor_instance
(
&
instance
->
options
,
instance
->
get_map
());
return
0
;
return
0
;
}
}
C_MODE_END
C_MODE_END
/*
Wait for an instance
SYNOPSYS
wait_process()
pi Pointer to the process information structure
(platform-dependent).
RETURN
0 - Success
1 - Error
*/
#ifndef __WIN__
static
int
wait_process
(
My_process_info
*
pi
)
{
/*
Here we wait for the child created. This process differs for systems
running LinuxThreads and POSIX Threads compliant systems. This is because
according to POSIX we could wait() for a child in any thread of the
process. While LinuxThreads require that wait() is called by the thread,
which created the child.
On the other hand we could not expect mysqld to return the pid, we
got in from fork(), to wait4() fucntion when running on LinuxThreads.
This is because MySQL shutdown thread is not the one, which was created
by our fork() call.
So basically we have two options: whether the wait() call returns only in
the creator thread, but we cannot use waitpid() since we have no idea
which pid we should wait for (in fact it should be the pid of shutdown
thread, but we don't know this one). Or we could use waitpid(), but
couldn't use wait(), because it could return in any wait() in the program.
*/
if
(
linuxthreads
)
wait
(
NULL
);
/* LinuxThreads were detected */
else
waitpid
(
*
pi
,
NULL
,
0
);
return
0
;
}
#else
static
int
wait_process
(
My_process_info
*
pi
)
{
/* Wait until child process exits. */
WaitForSingleObject
(
pi
->
hProcess
,
INFINITE
);
DWORD
exitcode
;
::
GetExitCodeProcess
(
pi
->
hProcess
,
&
exitcode
);
/* Close process and thread handles. */
CloseHandle
(
pi
->
hProcess
);
CloseHandle
(
pi
->
hThread
);
/*
GetExitCodeProces returns zero on failure. We should revert this value
to report an error.
*/
return
(
!
exitcode
);
}
#endif
/*
Launch an instance
SYNOPSYS
start_process()
instance_options Pointer to the options of the instance to be
launched.
pi Pointer to the process information structure
(platform-dependent).
RETURN
0 - Success
1 - Cannot create an instance
*/
#ifndef __WIN__
static
int
start_process
(
Instance_options
*
instance_options
,
My_process_info
*
pi
)
{
*
pi
=
fork
();
switch
(
*
pi
)
{
case
0
:
execv
(
instance_options
->
mysqld_path
,
instance_options
->
argv
);
/* exec never returns */
exit
(
1
);
case
1
:
log_info
(
"cannot fork() to start instance %s"
,
instance_options
->
instance_name
);
return
1
;
}
return
0
;
}
#else
static
int
start_process
(
Instance_options
*
instance_options
,
My_process_info
*
pi
)
{
STARTUPINFO
si
;
ZeroMemory
(
&
si
,
sizeof
(
STARTUPINFO
));
si
.
cb
=
sizeof
(
STARTUPINFO
);
ZeroMemory
(
pi
,
sizeof
(
PROCESS_INFORMATION
));
int
cmdlen
=
0
;
for
(
int
i
=
1
;
instance_options
->
argv
[
i
]
!=
0
;
i
++
)
cmdlen
+=
strlen
(
instance_options
->
argv
[
i
])
+
1
;
cmdlen
++
;
/* we have to add a single space for CreateProcess (see docs) */
char
*
cmdline
=
NULL
;
if
(
cmdlen
>
0
)
{
cmdline
=
new
char
[
cmdlen
];
cmdline
[
0
]
=
0
;
for
(
int
i
=
1
;
instance_options
->
argv
[
i
]
!=
0
;
i
++
)
{
strcat
(
cmdline
,
" "
);
strcat
(
cmdline
,
instance_options
->
argv
[
i
]);
}
}
/* Start the child process */
BOOL
result
=
CreateProcess
(
instance_options
->
mysqld_path
,
/* File to execute */
cmdline
,
/* Command line */
NULL
,
/* Process handle not inheritable */
NULL
,
/* Thread handle not inheritable */
FALSE
,
/* Set handle inheritance to FALSE */
0
,
/* No creation flags */
NULL
,
/* Use parent's environment block */
NULL
,
/* Use parent's starting directory */
&
si
,
/* Pointer to STARTUPINFO structure */
pi
);
/* Pointer to PROCESS_INFORMATION structure */
delete
cmdline
;
return
(
!
result
);
}
#endif
/*
Fork child, exec an instance and monitor it.
SYNOPSYS
start_and_monitor_instance()
old_instance_options Pointer to the options of the instance to be
launched. This info is likely to become obsolete
when function returns from wait_process()
instance_map Pointer to the instance_map. We use it to protect
the instance from deletion, while we are working
with it.
DESCRIPTION
Fork a child, then exec and monitor it. When the child is dead,
find appropriate instance (for this purpose we save its name),
set appropriate flags and wake all threads waiting for instance
to stop.
RETURN
Function returns no value
*/
static
void
start_and_monitor_instance
(
Instance_options
*
old_instance_options
,
Instance_map
*
instance_map
)
{
enum
{
MAX_INSTANCE_NAME_LEN
=
512
};
char
instance_name_buff
[
MAX_INSTANCE_NAME_LEN
];
uint
instance_name_len
;
Instance
*
current_instance
;
My_process_info
process_info
;
/*
Lock instance map to guarantee that no instances are deleted during
strmake() and execv() calls.
*/
instance_map
->
lock
();
/*
Save the instance name in the case if Instance object we
are using is destroyed. (E.g. by "FLUSH INSTANCES")
*/
strmake
(
instance_name_buff
,
old_instance_options
->
instance_name
,
MAX_INSTANCE_NAME_LEN
-
1
);
instance_name_len
=
old_instance_options
->
instance_name_len
;
log_info
(
"starting instance %s"
,
instance_name_buff
);
if
(
start_process
(
old_instance_options
,
&
process_info
))
return
;
/* error is logged */
/* allow users to delete instances */
instance_map
->
unlock
();
/* don't check for return value */
wait_process
(
&
process_info
);
current_instance
=
instance_map
->
find
(
instance_name_buff
,
instance_name_len
);
if
(
current_instance
)
current_instance
->
set_crash_flag_n_wake_all
();
return
;
}
Instance_map
*
Instance
::
get_map
()
{
return
instance_map
;
}
void
Instance
::
remove_pid
()
void
Instance
::
remove_pid
()
{
{
int
pid
;
int
pid
;
...
@@ -65,6 +286,7 @@ void Instance::remove_pid()
...
@@ -65,6 +286,7 @@ void Instance::remove_pid()
options
.
instance_name
);
options
.
instance_name
);
}
}
/*
/*
The method starts an instance.
The method starts an instance.
...
@@ -116,107 +338,24 @@ int Instance::start()
...
@@ -116,107 +338,24 @@ int Instance::start()
return
ER_INSTANCE_ALREADY_STARTED
;
return
ER_INSTANCE_ALREADY_STARTED
;
}
}
#ifndef __WIN__
/*
int
Instance
::
launch_and_wait
()
The method sets the crash flag and wakes all waiters on
{
COND_instance_stopped and COND_guardian
pid_t
pid
=
fork
();
switch
(
pid
)
{
case
0
:
execv
(
options
.
mysqld_path
,
options
.
argv
);
/* exec never returns */
exit
(
1
);
case
-
1
:
log_info
(
"cannot fork() to start instance %s"
,
options
.
instance_name
);
return
-
1
;
default:
/*
Here we wait for the child created. This process differs for systems
running LinuxThreads and POSIX Threads compliant systems. This is because
according to POSIX we could wait() for a child in any thread of the
process. While LinuxThreads require that wait() is called by the thread,
which created the child.
On the other hand we could not expect mysqld to return the pid, we
got in from fork(), to wait4() fucntion when running on LinuxThreads.
This is because MySQL shutdown thread is not the one, which was created
by our fork() call.
So basically we have two options: whether the wait() call returns only in
the creator thread, but we cannot use waitpid() since we have no idea
which pid we should wait for (in fact it should be the pid of shutdown
thread, but we don't know this one). Or we could use waitpid(), but
couldn't use wait(), because it could return in any wait() in the program.
*/
if
(
linuxthreads
)
wait
(
NULL
);
/* LinuxThreads were detected */
else
waitpid
(
pid
,
NULL
,
0
);
}
return
0
;
}
#else
int
Instance
::
launch_and_wait
()
{
STARTUPINFO
si
;
PROCESS_INFORMATION
pi
;
ZeroMemory
(
&
si
,
sizeof
(
si
));
si
.
cb
=
sizeof
(
si
);
ZeroMemory
(
&
pi
,
sizeof
(
pi
));
int
cmdlen
=
0
;
for
(
int
i
=
1
;
options
.
argv
[
i
]
!=
0
;
i
++
)
cmdlen
+=
strlen
(
options
.
argv
[
i
])
+
1
;
cmdlen
++
;
// we have to add a single space for CreateProcess (read the docs)
char
*
cmdline
=
NULL
;
if
(
cmdlen
>
0
)
{
cmdline
=
new
char
[
cmdlen
];
cmdline
[
0
]
=
0
;
for
(
int
i
=
1
;
options
.
argv
[
i
]
!=
0
;
i
++
)
{
strcat
(
cmdline
,
" "
);
strcat
(
cmdline
,
options
.
argv
[
i
]);
}
}
// Start the child process.
BOOL
result
=
CreateProcess
(
options
.
mysqld_path
,
// file to execute
cmdline
,
// Command line.
NULL
,
// Process handle not inheritable.
NULL
,
// Thread handle not inheritable.
FALSE
,
// Set handle inheritance to FALSE.
0
,
// No creation flags.
NULL
,
// Use parent's environment block.
NULL
,
// Use parent's starting directory.
&
si
,
// Pointer to STARTUPINFO structure.
&
pi
);
// Pointer to PROCESS_INFORMATION structure.
delete
cmdline
;
if
(
!
result
)
return
-
1
;
// Wait until child process exits.
WaitForSingleObject
(
pi
.
hProcess
,
INFINITE
);
DWORD
exitcode
;
::
GetExitCodeProcess
(
pi
.
hProcess
,
&
exitcode
);
// Close process and thread handles.
SYNOPSYS
CloseHandle
(
pi
.
hProcess
);
set_crash_flag_n_wake_all()
CloseHandle
(
pi
.
hThread
);
return
exitcode
;
DESCRIPTION
}
The method is called when an instance is crashed or terminated.
#endif
In the former case it might indicate that guardian probably should
restart it.
RETURN
Function returns no value
*/
void
Instance
::
fork_and_monitor
()
void
Instance
::
set_crash_flag_n_wake_all
()
{
{
log_info
(
"starting instance %s"
,
options
.
instance_name
);
if
(
launch_and_wait
())
return
;
/* error is logged */
/* set instance state to crashed */
/* set instance state to crashed */
pthread_mutex_lock
(
&
LOCK_instance
);
pthread_mutex_lock
(
&
LOCK_instance
);
crashed
=
1
;
crashed
=
1
;
...
@@ -230,11 +369,10 @@ void Instance::fork_and_monitor()
...
@@ -230,11 +369,10 @@ void Instance::fork_and_monitor()
pthread_cond_signal
(
&
COND_instance_stopped
);
pthread_cond_signal
(
&
COND_instance_stopped
);
/* wake guardian */
/* wake guardian */
pthread_cond_signal
(
&
instance_map
->
guardian
->
COND_guardian
);
pthread_cond_signal
(
&
instance_map
->
guardian
->
COND_guardian
);
/* thread exits */
return
;
}
}
Instance
::
Instance
()
:
crashed
(
0
)
Instance
::
Instance
()
:
crashed
(
0
)
{
{
pthread_mutex_init
(
&
LOCK_instance
,
0
);
pthread_mutex_init
(
&
LOCK_instance
,
0
);
...
...
server-tools/instance-manager/instance.h
View file @
5605117d
...
@@ -41,7 +41,8 @@ class Instance
...
@@ -41,7 +41,8 @@ class Instance
/* send a signal to the instance */
/* send a signal to the instance */
void
kill_instance
(
int
signo
);
void
kill_instance
(
int
signo
);
int
is_crashed
();
int
is_crashed
();
void
fork_and_monitor
();
void
set_crash_flag_n_wake_all
();
Instance_map
*
Instance
::
get_map
();
public:
public:
enum
{
DEFAULT_SHUTDOWN_DELAY
=
35
};
enum
{
DEFAULT_SHUTDOWN_DELAY
=
35
};
...
@@ -63,7 +64,6 @@ class Instance
...
@@ -63,7 +64,6 @@ class Instance
Instance_map
*
instance_map
;
Instance_map
*
instance_map
;
void
remove_pid
();
void
remove_pid
();
int
launch_and_wait
();
};
};
#endif
/* INCLUDES_MYSQL_INSTANCE_MANAGER_INSTANCE_H */
#endif
/* INCLUDES_MYSQL_INSTANCE_MANAGER_INSTANCE_H */
server-tools/instance-manager/instance_options.cc
View file @
5605117d
...
@@ -130,6 +130,8 @@ int Instance_options::fill_instance_version()
...
@@ -130,6 +130,8 @@ int Instance_options::fill_instance_version()
version_option
,
sizeof
(
version_option
)))
version_option
,
sizeof
(
version_option
)))
goto
err
;
goto
err
;
bzero
(
result
,
MAX_VERSION_STRING_LENGTH
);
rc
=
parse_output_and_get_value
(
cmd
.
buffer
,
mysqld_path
,
rc
=
parse_output_and_get_value
(
cmd
.
buffer
,
mysqld_path
,
result
,
MAX_VERSION_STRING_LENGTH
,
result
,
MAX_VERSION_STRING_LENGTH
,
GET_LINE
);
GET_LINE
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment