about summary refs log tree commit diff
diff options
context:
space:
mode:
-rw-r--r--README.md1
-rw-r--r--lr.c35
2 files changed, 36 insertions, 0 deletions
diff --git a/README.md b/README.md
index afa9c17..6cbf19f 100644
--- a/README.md
+++ b/README.md
@@ -112,6 +112,7 @@ Default: `n`.
 * `p`: directory name.
 * `s`: file size.
 * `t`: file type.  This sorts all directories before other files.
+* `v`: file name as version numbers (sorts "2" before "10").
 
 ## Filter expressions
 
diff --git a/lr.c b/lr.c
index bc91f8c..b815ddd 100644
--- a/lr.c
+++ b/lr.c
@@ -1040,9 +1040,42 @@ dircmp(char *a, char *b)
 	return *a - *b;
 }
 
+// taken straight from musl@a593414
+int mystrverscmp(const char *l0, const char *r0)
+{
+	const unsigned char *l = (const void *)l0;
+	const unsigned char *r = (const void *)r0;
+	size_t i, dp, j;
+	int z = 1;
+
+	/* Find maximal matching prefix and track its maximal digit
+	 * suffix and whether those digits are all zeros. */
+	for (dp=i=0; l[i]==r[i]; i++) {
+		int c = l[i];
+		if (!c) return 0;
+		if (!isdigit(c)) dp=i+1, z=1;
+		else if (c!='0') z=0;
+	}
+
+	if (l[dp]!='0' && r[dp]!='0') {
+		/* If we're not looking at a digit sequence that began
+		 * with a zero, longest digit string is greater. */
+		for (j=i; isdigit(l[j]); j++)
+			if (!isdigit(r[j])) return 1;
+		if (isdigit(r[j])) return -1;
+	} else if (z && dp<i && (isdigit(l[i]) || isdigit(r[i]))) {
+		/* Otherwise, if common prefix of digit sequence is
+		 * all zeros, digits order less than non-digits. */
+		return (unsigned char)(l[i]-'0') - (unsigned char)(r[i]-'0');
+	}
+
+	return l[i] - r[i];
+}
+
 #define CMP(a, b) if ((a) == (b)) break; else if ((a) < (b)) return -1; else return 1
 #define STRCMP(a, b) if (strcmp(a, b) == 0) break; else return (strcmp(a, b));
 #define DIRCMP(a, b) { int r = dircmp(a, b); if (r == 0) break; else return r; };
+#define VERCMP(a, b) if (mystrverscmp(a, b) == 0) break; else return (mystrverscmp(a, b));
 
 int
 order(const void *a, const void *b)
@@ -1076,6 +1109,8 @@ order(const void *a, const void *b)
 		case 'E': STRCMP(extnam(fb->fpath), extnam(fa->fpath));
 		case 'p': DIRCMP(fa->fpath, fb->fpath);
 		case 'P': DIRCMP(fb->fpath, fa->fpath);
+		case 'v': VERCMP(fa->fpath, fb->fpath);
+		case 'V': VERCMP(fb->fpath, fa->fpath);
 		default: STRCMP(fa->fpath, fb->fpath);
 		}
 	}