From 59283b6e3a1fe24f25d1a339dc2e824bd2e4cc2d Mon Sep 17 00:00:00 2001 From: Fletcher Dostie Date: Fri, 1 Sep 2023 18:02:22 -0700 Subject: [PATCH 1/6] Check whether library can be loaded before returning it --- pylink/library.py | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/pylink/library.py b/pylink/library.py index df50a41..8c31183 100644 --- a/pylink/library.py +++ b/pylink/library.py @@ -173,6 +173,22 @@ def find_library_windows(cls): if os.path.isfile(lib_path): yield lib_path + @classmethod + def can_load_library(dllpath): + """Test whether a library is the correct architecture to load. + + Args: + dllpath: A path to a library. + + Returns: + True if the library could be successfully loaded, False if not. + """ + try: + ctypes.CDLL(dllpath) + return True + except: + return False + @classmethod def find_library_linux(cls): """Loads the SEGGER DLL from the root directory. @@ -202,6 +218,8 @@ def find_library_linux(cls): for fname in fnames: fpath = os.path.join(directory_name, fname) + if not self.can_load_library(fpath): + continue if util.is_os_64bit(): if '_x86' not in fname: yield fpath From 23ef04e2ae04aa265b331d4c03bdd15984f1eb87 Mon Sep 17 00:00:00 2001 From: Fletcher Dostie Date: Fri, 1 Sep 2023 18:06:01 -0700 Subject: [PATCH 2/6] Check whether library can be loaded before returning it --- pylink/library.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pylink/library.py b/pylink/library.py index 8c31183..71b7782 100644 --- a/pylink/library.py +++ b/pylink/library.py @@ -218,7 +218,7 @@ def find_library_linux(cls): for fname in fnames: fpath = os.path.join(directory_name, fname) - if not self.can_load_library(fpath): + if not Library.can_load_library(fpath): continue if util.is_os_64bit(): if '_x86' not in fname: From 9481ae09e864da0ad206a82d4b447ae533535e69 Mon Sep 17 00:00:00 2001 From: Fletcher Dostie Date: Fri, 1 Sep 2023 18:09:25 -0700 Subject: [PATCH 3/6] Check whether library can be loaded before returning it --- pylink/library.py | 34 +++++++++++++++++----------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/pylink/library.py b/pylink/library.py index 71b7782..5fef43e 100644 --- a/pylink/library.py +++ b/pylink/library.py @@ -138,6 +138,22 @@ def get_appropriate_windows_sdk_name(cls): else: return Library.WINDOWS_32_JLINK_SDK_NAME + @classmethod + def can_load_library(dllpath): + """Test whether a library is the correct architecture to load. + + Args: + dllpath: A path to a library. + + Returns: + True if the library could be successfully loaded, False if not. + """ + try: + ctypes.CDLL(dllpath) + return True + except: + return False + @classmethod def find_library_windows(cls): """Loads the SEGGER DLL from the windows installation directory. @@ -173,22 +189,6 @@ def find_library_windows(cls): if os.path.isfile(lib_path): yield lib_path - @classmethod - def can_load_library(dllpath): - """Test whether a library is the correct architecture to load. - - Args: - dllpath: A path to a library. - - Returns: - True if the library could be successfully loaded, False if not. - """ - try: - ctypes.CDLL(dllpath) - return True - except: - return False - @classmethod def find_library_linux(cls): """Loads the SEGGER DLL from the root directory. @@ -218,7 +218,7 @@ def find_library_linux(cls): for fname in fnames: fpath = os.path.join(directory_name, fname) - if not Library.can_load_library(fpath): + if not cls.can_load_library(fpath): continue if util.is_os_64bit(): if '_x86' not in fname: From 43f3c01ce37c25e0211da602d9d56a9c35b9be0b Mon Sep 17 00:00:00 2001 From: Fletcher Dostie Date: Fri, 1 Sep 2023 18:10:38 -0700 Subject: [PATCH 4/6] Check whether library can be loaded before returning it --- pylink/library.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pylink/library.py b/pylink/library.py index 5fef43e..6def254 100644 --- a/pylink/library.py +++ b/pylink/library.py @@ -139,7 +139,7 @@ def get_appropriate_windows_sdk_name(cls): return Library.WINDOWS_32_JLINK_SDK_NAME @classmethod - def can_load_library(dllpath): + def can_load_library(cls, dllpath): """Test whether a library is the correct architecture to load. Args: From e73700f706c3fd965e2a9e288e8639cf858837f4 Mon Sep 17 00:00:00 2001 From: Fletcher Dostie Date: Wed, 13 Sep 2023 13:32:08 -0700 Subject: [PATCH 5/6] Catch OSError instead of raw exception; Do check after 64bit OS check --- pylink/library.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/pylink/library.py b/pylink/library.py index 6def254..6361abf 100644 --- a/pylink/library.py +++ b/pylink/library.py @@ -151,7 +151,7 @@ def can_load_library(cls, dllpath): try: ctypes.CDLL(dllpath) return True - except: + except OSError: return False @classmethod @@ -218,10 +218,10 @@ def find_library_linux(cls): for fname in fnames: fpath = os.path.join(directory_name, fname) - if not cls.can_load_library(fpath): - continue if util.is_os_64bit(): - if '_x86' not in fname: + if not cls.can_load_library(fpath): + continue + elif '_x86' not in fname: yield fpath elif x86_found: if '_x86' in fname: From 68df01998b1a2b6c3ea7ad6d3fb7edd8c0af40ed Mon Sep 17 00:00:00 2001 From: Fletcher Dostie Date: Thu, 11 Jul 2024 17:10:21 -0700 Subject: [PATCH 6/6] Incorporate unit test from https://github.com/square/pylink/pull/182 --- pylink/library.py | 5 ++-- tests/unit/test_library.py | 57 ++++++++++++++++++++++++++++++++++++-- 2 files changed, 57 insertions(+), 5 deletions(-) diff --git a/pylink/library.py b/pylink/library.py index 6361abf..ca4d33f 100644 --- a/pylink/library.py +++ b/pylink/library.py @@ -143,10 +143,11 @@ def can_load_library(cls, dllpath): """Test whether a library is the correct architecture to load. Args: - dllpath: A path to a library. + cls (Library): the ``Library`` class + dllpath (str): A path to a library. Returns: - True if the library could be successfully loaded, False if not. + ``True`` if the library could be successfully loaded, ``False`` if not. """ try: ctypes.CDLL(dllpath) diff --git a/tests/unit/test_library.py b/tests/unit/test_library.py index fb92f92..e12f967 100644 --- a/tests/unit/test_library.py +++ b/tests/unit/test_library.py @@ -880,13 +880,16 @@ def test_linux_6_10_0_32bit(self, mock_os, mock_load_library, mock_find_library, @mock.patch('tempfile.NamedTemporaryFile', new=mock.Mock()) @mock.patch('ctypes.util.find_library') @mock.patch('ctypes.cdll.LoadLibrary') + @mock.patch('ctypes.CDLL') @mock.patch('pylink.library.os') - def test_linux_6_10_0_64bit(self, mock_os, mock_load_library, mock_find_library, mock_open, mock_is_os_64bit): + def test_linux_6_10_0_64bit(self, mock_os, mock_cdll, mock_load_library, + mock_find_library, mock_open, mock_is_os_64bit): """Tests finding the DLL on Linux through the SEGGER application for V6.0.0+ on 64 bit linux. Args: self (TestLibrary): the ``TestLibrary`` instance mock_os (Mock): a mocked version of the ``os`` module + mock_cdll (Mock): a mocked version of the `cdll.CDLL` class constructor mock_load_library (Mock): a mocked version of the library loader mock_find_library (Mock): a mocked call to ``ctypes`` find library mock_open (Mock): mock for mocking the call to ``open()`` @@ -896,6 +899,7 @@ def test_linux_6_10_0_64bit(self, mock_os, mock_load_library, mock_find_library, ``None`` """ mock_find_library.return_value = None + mock_cdll.return_value = None directories = [ '/opt/SEGGER/JLink_Linux_V610d_x86_64/libjlinkarm_x86.so.6.10', '/opt/SEGGER/JLink_Linux_V610d_x86_64/libjlinkarm.so.6.10', @@ -918,6 +922,49 @@ def test_linux_6_10_0_64bit(self, mock_os, mock_load_library, mock_find_library, lib.unload = mock.Mock() self.assertEqual(None, lib._path) + @mock.patch('sys.platform', new='linux2') + @mock.patch('pylink.util.is_os_64bit', return_value=True) + @mock.patch('pylink.library.open') + @mock.patch('os.remove', new=mock.Mock()) + @mock.patch('tempfile.NamedTemporaryFile', new=mock.Mock()) + @mock.patch('ctypes.util.find_library') + @mock.patch('ctypes.cdll.LoadLibrary') + @mock.patch('ctypes.CDLL') + @mock.patch('pylink.library.os') + def test_linux_64bit_no_x86(self, mock_os, mock_cdll, mock_load_library, + mock_find_library, mock_open, mock_is_os_64bit): + """Tests finding the DLL on Linux when no library name contains 'x86'. + + Args: + self (TestLibrary): the ``TestLibrary`` instance + mock_os (Mock): a mocked version of the ``os`` module + mock_cdll (Mock): a mocked version of the `cdll.CDLL` class constructor + mock_load_library (Mock): a mocked version of the library loader + mock_find_library (Mock): a mocked call to ``ctypes`` find library + mock_open (Mock): mock for mocking the call to ``open()`` + mock_is_os_64bit (Mock): mock for mocking the call to ``is_os_64bit``, returns True + + Returns: + ``None`` + """ + def on_cdll(name): + if '_arm' in name: + raise OSError + + mock_find_library.return_value = None + mock_cdll.side_effect = on_cdll + directories = [ + '/opt/SEGGER/JLink_Linux_V610d_x86_64/libjlinkarm_arm.so.6.10', + '/opt/SEGGER/JLink_Linux_V610d_x86_64/libjlinkarm.so.6.10', + ] + + self.mock_directories(mock_os, directories, '/') + + lib = library.Library() + lib.unload = mock.Mock() + load_library_args, load_libary_kwargs = mock_load_library.call_args + self.assertEqual(directories[1], lib._path) + @mock.patch('sys.platform', new='linux') @mock.patch('pylink.library.open') @mock.patch('os.remove', new=mock.Mock()) @@ -960,8 +1007,9 @@ def test_linux_empty(self, mock_os, mock_load_library, mock_find_library, mock_o @mock.patch('pylink.platform.libc_ver', return_value=('libc', '1.0')) @mock.patch('ctypes.util.find_library', return_value='libjlinkarm.so.7') @mock.patch('pylink.library.JLinkarmDlInfo.__init__') + @mock.patch('ctypes.CDLL') @mock.patch('ctypes.cdll.LoadLibrary') - def test_linux_glibc_unavailable(self, mock_load_library, mock_dlinfo_ctr, mock_find_library, + def test_linux_glibc_unavailable(self, mock_load_library, mock_cdll, mock_dlinfo_ctr, mock_find_library, mock_libc_ver, mock_is_os_64bit, mock_os, mock_open): """Confirms the whole JLinkarmDlInfo code path is not involved when GNU libc extensions are unavailable on a Linux system, and that we'll successfully fallback @@ -974,6 +1022,7 @@ def test_linux_glibc_unavailable(self, mock_load_library, mock_dlinfo_ctr, mock_ to the "search by file name" code path, aka find_library_linux() - and "successfully load" a mock library file from /opt/SEGGER/JLink """ + mock_cdll.side_effect = None directories = [ # Library.find_library_linux() should find this. '/opt/SEGGER/JLink/libjlinkarm.so.6' @@ -999,8 +1048,9 @@ def test_linux_glibc_unavailable(self, mock_load_library, mock_dlinfo_ctr, mock_ @mock.patch('pylink.util.is_os_64bit', return_value=True) @mock.patch('pylink.platform.libc_ver', return_value=('glibc', '2.34')) @mock.patch('ctypes.util.find_library') + @mock.patch('ctypes.CDLL') @mock.patch('ctypes.cdll.LoadLibrary') - def test_linux_dl_unavailable(self, mock_load_library, mock_find_library, mock_libc_ver, + def test_linux_dl_unavailable(self, mock_load_library, mock_cdll, mock_find_library, mock_libc_ver, mock_is_os_64bit, mock_os, mock_open): """Confirms we successfully fallback to the "search by file name" code path when libdl is unavailable despite the host system presenting itself as POSIX (GNU/Linux). @@ -1012,6 +1062,7 @@ def test_linux_dl_unavailable(self, mock_load_library, mock_find_library, mock_l to the "search by file name" code path, aka find_library_linux() - and "successfully load" a mock library file from /opt/SEGGER/JLink """ + mock_cdll.side_effect = None mock_find_library.side_effect = [ # find_library('jlinkarm') 'libjlinkarm.so.6',