blob: 605fb9b8303fc4abc68b25a33dfd79f3be0be9c4 [file] [log] [blame]
Derek Beckett4e928f62020-09-17 09:57:01 -07001# Lint as: python2, python3
apw2366d992007-03-12 20:35:57 +00002#
3# kernel_versions.py -- linux kernel version comparisons
4#
Derek Beckett4e928f62020-09-17 09:57:01 -07005from __future__ import absolute_import
6from __future__ import division
7from __future__ import print_function
apw2366d992007-03-12 20:35:57 +00008__author__ = """Copyright Andy Whitcroft 2007"""
9
10import sys,re
11
Derek Beckett4e928f62020-09-17 09:57:01 -070012from six.moves import range
13
jadmanski0afbb632008-06-06 21:10:57 +000014#
apw2366d992007-03-12 20:35:57 +000015# Sort key for ordering versions chronologically. The key ordering
16# problem is between that introduced by -rcN. These come _before_
17# their accompanying version.
jadmanski0afbb632008-06-06 21:10:57 +000018#
apw2366d992007-03-12 20:35:57 +000019# 2.6.0 -> 2.6.1-rc1 -> 2.6.1
jadmanski0afbb632008-06-06 21:10:57 +000020#
apw2366d992007-03-12 20:35:57 +000021# In order to sort them we convert all non-rc releases to a pseudo
22# -rc99 release. We also convert all numbers to two digits. The
23# result is then sortable textually.
jadmanski0afbb632008-06-06 21:10:57 +000024#
apw2366d992007-03-12 20:35:57 +000025# 02.06.00-rc99 -> 02.06.01-rc01 -> 02.06.01-rc99
jadmanski0afbb632008-06-06 21:10:57 +000026#
mblighba07f6d2008-06-05 22:38:11 +000027encode_sep = re.compile(r'(\D+)')
28
apw2366d992007-03-12 20:35:57 +000029def version_encode(version):
jadmanski0afbb632008-06-06 21:10:57 +000030 bits = encode_sep.split(version)
31 n = 9
32 if len(bits[0]) == 0:
33 n += 2
34 if len(bits) == n or (len(bits) > n and bits[n] != '_rc'):
35 # Insert missing _rc99 after 2 . 6 . 18 -smp- 220 . 0
36 bits.insert(n, '_rc')
37 bits.insert(n+1, '99')
38 n = 5
39 if len(bits[0]) == 0:
40 n += 2
41 if len(bits) <= n or bits[n] != '-rc':
42 bits.insert(n, '-rc')
43 bits.insert(n+1, '99')
44 for n in range(0, len(bits), 2):
45 if len(bits[n]) == 1:
46 bits[n] = '0' + bits[n]
apw2366d992007-03-12 20:35:57 +000047
jadmanski0afbb632008-06-06 21:10:57 +000048 return ''.join(bits)
apw2366d992007-03-12 20:35:57 +000049
50
51def version_limit(version, n):
jadmanski0afbb632008-06-06 21:10:57 +000052 bits = encode_sep.split(version)
53 return ''.join(bits[0:n])
apw2366d992007-03-12 20:35:57 +000054
55
56def version_len(version):
jadmanski0afbb632008-06-06 21:10:57 +000057 return len(encode_sep.split(version))
apw2366d992007-03-12 20:35:57 +000058
59#
60# Given a list of versions find the nearest version which is deemed
61# less than or equal to the target. Versions are in linux order
62# as follows:
jadmanski0afbb632008-06-06 21:10:57 +000063#
apw2366d992007-03-12 20:35:57 +000064# 2.6.0 -> 2.6.1 -> 2.6.2-rc1 -> 2.6.2-rc2 -> 2.6.2 -> 2.6.3-rc1
65# | |\
66# | | 2.6.2-rc1-mm1 -> 2.6.2-rc1-mm2
67# | \
68# | 2.6.2-rc1-ac1 -> 2.6.2-rc1-ac2
jadmanski0afbb632008-06-06 21:10:57 +000069# \
apw2366d992007-03-12 20:35:57 +000070# 2.6.1-mm1 -> 2.6.1-mm2
jadmanski0afbb632008-06-06 21:10:57 +000071#
apw2366d992007-03-12 20:35:57 +000072# Note that a 2.6.1-mm1 is not a predecessor of 2.6.2-rc1-mm1.
73#
74def version_choose_config(version, candidates):
jadmanski0afbb632008-06-06 21:10:57 +000075 # Check if we have an exact match ... if so magic
76 if version in candidates:
77 return version
apw2366d992007-03-12 20:35:57 +000078
jadmanski0afbb632008-06-06 21:10:57 +000079 # Sort the search key into the list ordered by 'age'
80 deco = [ (version_encode(v), i, v) for i, v in
81 enumerate(candidates + [ version ]) ]
82 deco.sort()
83 versions = [ v for _, _, v in deco ]
apw2366d992007-03-12 20:35:57 +000084
jadmanski0afbb632008-06-06 21:10:57 +000085 # Everything sorted below us is of interst.
86 for n in range(len(versions) - 1, -1, -1):
87 if versions[n] == version:
88 break
89 n -= 1
apw2366d992007-03-12 20:35:57 +000090
jadmanski0afbb632008-06-06 21:10:57 +000091 # Try ever shorter 'prefixes' 2.6.20-rc3-mm, 2.6.20-rc, 2.6. etc
92 # to match against the ordered list newest to oldest.
93 length = version_len(version) - 1
94 version = version_limit(version, length)
95 while length > 1:
96 for o in range(n, -1, -1):
97 if version_len(versions[o]) == (length + 1) and \
mbligh8b352852008-06-07 01:07:08 +000098 version_limit(versions[o], length) == version:
jadmanski0afbb632008-06-06 21:10:57 +000099 return versions[o]
100 length -= 2
101 version = version_limit(version, length)
apw2366d992007-03-12 20:35:57 +0000102
jadmanski0afbb632008-06-06 21:10:57 +0000103 return None
mblighba07f6d2008-06-05 22:38:11 +0000104
105
106def is_released_kernel(version):
jadmanski0afbb632008-06-06 21:10:57 +0000107 # True if version name suggests a released kernel,
108 # not some release candidate or experimental kernel name
109 # e.g. 2.6.18-smp-200.0 includes no other text, underscores, etc
110 version = version.strip('01234567890.-')
111 return version in ['', 'smp', 'smpx', 'pae']
mblighba07f6d2008-06-05 22:38:11 +0000112
113
114def is_release_candidate(version):
jadmanski0afbb632008-06-06 21:10:57 +0000115 # True if version names a released kernel or release candidate,
116 # not some experimental name containing arbitrary text
117 # e.g. 2.6.18-smp-220.0_rc3 but not 2.6.18_patched
118 version = re.sub(r'[_-]rc\d+', '', version)
119 return is_released_kernel(version)